使用 GregorianCalendar 比较日期

Compare Date using GregorianCalendar

我正在尝试将当前日期 (2016-08-31) 与给定日期 (2016-08-31) 进行比较。 我当前的移动设备时区是太平洋时间 GMT-08:00。

如果我在设备上禁用自动日期和时区并将时区设置为 GMT+08:00 珀斯,方法 1 将 return 为真,但方法 2 return 为假;

方法2的结果是预期的,因为我比较的是不带时区的日期,所以“2016-08-31”之前的“2​​016-08-31”是假的;为什么方法 1 return 是正确的?

    public boolean method1() {
        try {
            GregorianCalendar currentCalendar = new GregorianCalendar();
            SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
            Date endDate = sdf.parse("2016-08-31");
            Calendar endCalendar = new GregorianCalendar();
            endCalendar.setTime(endDate);

            if (endCalendar.before(currentCalendar)) {
                return true;
            } else {
                return false;
            }
        } catch (ParseException e) {
            ...
        }
    }


    public boolean method2() {    
        try {
            SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
            Date currentDate = sdf.parse(formatter.format(new Date())); 
            Date endDate = sdf.parse("2016-08-31");

            if (endDate.before(currentDate)) {
                return true;
            } else {
                return false;
            }

        } catch (ParseException e) {
            ...
        }
    }

带时区和不带时区

可能的解释:您的代码混合使用分区和 UTC 日期时间对象。在某些行中,您为一个对象分配了一个时区 (java.util.Calendar),这是您的 JVM 当前的默认时区。在其他行中,您有一个固定在 UTC. The Australia/Perth 时区的对象 (java.util.Date) 比 UTC 早 8 小时,因此您当然可以看到日期差异。打印出他们的自纪元以来的毫秒数将使比较结果一目了然。

帮自己一个 忙:避免使用这些臭名昭著的麻烦旧 classes。请改用 java.time。

Boolean isFuture = 
    LocalDate.parse( "2016-08-31" )
             .isAfter( LocalDate.now( ZoneId.of( "Australia/Perth" ) ) ) ;

使用java.time

您正在使用麻烦的旧旧日期时间 classes,现在已被 java.time classes 取代。

获取当前日期需要时区。对于任何给定时刻,日期在全球范围内因地区而异。如果省略,则隐式应用 JVM 当前的默认时区。最好指定,因为该默认值随时可能更改。

指定 proper time zone name。切勿使用 3-4 个字母的缩写,例如 ESTIST,因为它们不是真正的时区,不是标准化的,甚至不是唯一的(!)。

ZoneId z = ZoneId.of( "Australia/Perth" ) ;

LocalDate class 表示没有时间和时区的仅日期值。

LocalDate today = LocalDate.now( z );

您输入的字符串恰好符合标准 ISO 8601 格式。所以直接用LocalDate解析。无需指定格式模式。

LocalDate ld = LocalDate.parse( "2016-08-31" );

比较isBefore,isAfter, isEqual, and compareTo.

Boolean isFuture = ld.isAfter( today );

关于java.time

java.time 框架内置于 Java 8 及更高版本中。这些 classes 取代了麻烦的旧日期时间 classes,例如 java.util.Date.Calendarjava.text.SimpleDateFormat

Joda-Time project, now in maintenance mode,建议迁移到java.time。

要了解更多信息,请参阅 Oracle Tutorial。并在 Stack Overflow 中搜索许多示例和解释。

许多 java.time 功能被反向移植到 ThreeTen-Backport and further adapted to Android in ThreeTenABP (see 中的 Java 6 & 7)。

ThreeTen-Extra 项目用额外的 class 扩展了 java.time。该项目是未来可能添加到 java.time 的试验场。您可能会在这里找到一些有用的 classes,例如 IntervalYearWeekYearQuarter 等。