首页 > 解决方案 > 一年中的一周的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>

标签: javahivesimpledateformatdatetime-formatweek-number

解决方案


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.ISOLocalDate

    DateTimeFormatter weekFormatter = DateTimeFormatter.ofPattern("ww", Locale.FRANCE);
    System.out.println(ld.format(weekFormatter));

输出是一样的:

42

传递给DateTimeFormatter.ofPattern确定周计划的语言环境。如果我通过Locale.US,我得到 43。

我建议您使用现代 Java 日期和时间 API java.time,并远离旧的日期时间类,如SimpleDateFormatCalendar。旧的设计不佳,而现代的则更好用。

你的代码出了什么问题?

过时的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 年的周数是一致的。


推荐阅读