使用 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”之前的“2016-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 个字母的缩写,例如 EST
或 IST
,因为它们不是真正的时区,不是标准化的,甚至不是唯一的(!)。
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
、.Calendar
和 java.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,例如 Interval
、YearWeek
、YearQuarter
等。
我正在尝试将当前日期 (2016-08-31) 与给定日期 (2016-08-31) 进行比较。 我当前的移动设备时区是太平洋时间 GMT-08:00。
如果我在设备上禁用自动日期和时区并将时区设置为 GMT+08:00 珀斯,方法 1 将 return 为真,但方法 2 return 为假;
方法2的结果是预期的,因为我比较的是不带时区的日期,所以“2016-08-31”之前的“2016-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 个字母的缩写,例如 EST
或 IST
,因为它们不是真正的时区,不是标准化的,甚至不是唯一的(!)。
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
、.Calendar
和 java.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
ThreeTen-Extra 项目用额外的 class 扩展了 java.time。该项目是未来可能添加到 java.time 的试验场。您可能会在这里找到一些有用的 classes,例如 Interval
、YearWeek
、YearQuarter
等。