java - 一年中的一周的Java日期库中的不一致
问题描述
根据https://en.wikipedia.org/wiki/ISO_8601#Week_dates,工作日从星期一开始。但是,从 Java 中,如果您尝试以两种不同的方式提取周数,如果日期是星期日,则会出现两个不同的输出。
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Calendar;
public class TestDate {
public static void main(String[] args) throws ParseException {
final SimpleDateFormat formatter = new SimpleDateFormat("yyyy-MM-dd");
final SimpleDateFormat weekFormatter = new SimpleDateFormat("ww");
String date = "2018-10-21";
System.out.println(weekFormatter.format(formatter.parse(date)));
Calendar calendar = Calendar.getInstance();
calendar.setFirstDayOfWeek(Calendar.MONDAY);
calendar.setTime(formatter.parse(date));
System.out.println(calendar.get(Calendar.WEEK_OF_YEAR));
}
}
输出:
43
42
这是不一致吗?
这只是我为重现问题而编写的一个测试程序,我注意到 Hive 中的问题,如下所示:
0: jdbc:hive2://zk0-something> select from_unixtime(t, 'ww'), weekofyear(from_unixtime(t, 'yyyy-MM-dd')) from (select 1540122033 as t) a;
+------+------+--+
| _c0 | _c1 |
+------+------+--+
| 43 | 42 |
+------+------+--+
1 row selected (0.388 seconds)
0: jdbc:hive2://zk0-something>
解决方案
java.time
String date = "2018-10-21";
LocalDate ld = LocalDate.parse(date);
int weekOfYear = ld.get(WeekFields.ISO.weekOfYear());
System.out.println(weekOfYear);
输出:
42
由于您对周数的ISO 8601规则感兴趣,因此可用于从. 如果您愿意,也可以使用格式化程序:WeekFields.ISO
LocalDate
DateTimeFormatter weekFormatter = DateTimeFormatter.ofPattern("ww", Locale.FRANCE);
System.out.println(ld.format(weekFormatter));
输出是一样的:
42
传递给DateTimeFormatter.ofPattern
确定周计划的语言环境。如果我通过Locale.US
,我得到 43。
我建议您使用现代 Java 日期和时间 API java.time,并远离旧的日期时间类,如SimpleDateFormat
和Calendar
。旧的设计不佳,而现代的则更好用。
你的代码出了什么问题?
过时的SimpleDateFormat
班级和现代班级都DateTimeFormatter
从他们的语言环境中获取他们的周编号方案。如果没有为格式化程序指定语言环境,它将使用 JVM 的默认语言环境。因此,如果 JVM 具有美国语言环境,例如,格式化程序将在您的第一个示例中打印 43,因为在美国,今年 10 月 21 日星期日是第 43 周。如果语言环境是法语,它将打印 42,因为那天是周42 在法国。法国遵循 ISO 8601 标准,美国没有。
在您的示例中,将Calendar
的第一天设置为星期一会导致周数为 42,如您所料。然而,情况并非总是如此。周数不仅由一周的第一天定义,而且由第 1 周的定义定义。从您的链接:
一年中的第一个 ISO 周可能最多有 3 天实际上是在即将结束的公历年中;如果是星期一、星期二和星期三。类似地,一年中的最后一个 ISO 周实际上可能在开始的公历年中最多有 3 天;如果是周五、周六和周日。每个 ISO 周的星期四总是在由 ISO 周编号年表示的公历年中。
美国对第 1 周的定义不同:在美国,1 月 1 日总是在第 1 周。因此,如果您Calendar
是使用美国语言环境创建的,则将其一周的第一天设置为星期一是不够的,遵循 ISO 8601 规则. 不过巧合的是,2018 年的周数是一致的。
推荐阅读
- javascript - 尝试链接过滤器选项选择
- java - 处理告警通知
- python - 让游戏中的蛇一点一点移动而不是整体移动
- cuda - PyCUDA mem_get_ipc_handle 给出 LogicError: cuIpcGetMemHandle failed: operation not supported
- python - 可能溢出?
- javascript - React js - componentDidMount() 代码是否作为 JavaScript 发送到客户端浏览器?
- angular - 以角度将小更改上传到生产服务器
- swift - [Swift[如何遍历对象的所有属性
- c# - Atata iframe SwitchTo 方法编译错误:使用泛型类型需要 1 个类型参数
- go - Golang获取当前文化的小数分隔符