使用Calendar格式化时间达不到预期的问题

Java8中文文档:http://www.matools.com/api/java8

java8 LocalDate参考:https://www.journaldev.com/2800/java-8-date-localdate-localdatetime-instant

背景:涉及到期限的地方如活动起止时间(2019-11-11 00:00:00~2019-11-18 00:00:00)、奖品有效时间(2019-12-1 00:00:00~2019-12-30 23:59:59)等,对时间部分做了格式化。

实现:

Calendar calendar = Calendar.getInstance();
 calendar.setTime(new Date());
 calendar.set(Calendar.HOUR_OF_DAY, 0);
 calendar.set(Calendar.MINUTE, 0);
 calendar.set(Calendar.MILLISECOND, 0);
 calendar.set(Calendar.SECOND, 0);
 return calendar.getTime();

问题:如下图所示,部分时间未按预期格式化:

使用Calendar格式化时间达不到预期的问题使用Calendar格式化时间达不到预期的问题

原因:格式化后,get操作出现误差(<1s),getTime的实现是根据毫秒数来的:

public final Date getTime() {
        return new Date(getTimeInMillis());
    }

改进: 由于我们需要的是一个确切的时间节点,而不关注该时间的变化与其包含的具体含义,所以我们可以选择java8新的日期时间Api LocalDate、LocalDateTime。

以下摘自互联网,为什么我们需要一个新的API?java中现有的日期和时间相关类存在一些问题,其中一些是:

  1. Java Date Time类没有一致定义,我们在包java.util和java.sql包中都有Date Class 。再次格式化和解析类在java.text包中定义。

  2. java.util.Date包含日期和时间,而java.sql.Date仅包含日期。java.sql包装中的这个没有意义。这两个类都有相同的名称,这本身就是一个非常糟糕的设计。

  3. 时间,时间戳,格式和解析没有明确定义的类。我们有java.text.DateFormat用于解析和格式化需求的抽象类。通常SimpleDateFormat类用于解析和格式化。

  4. 所有Date类都是可变的,因此它们不是线程安全的。这是Java Date和Calendar类的最大问题之一。

  5. 日期类不提供国际化,没有时区支持。所以java.util.Calendar和java.util.TimeZone类进行了介绍,但他们也有上面列出的所有问题。

实现:

LocalDate today = LocalDate.now();
LocalDateTime format = LocalDateTime.of(today.getYear(), today.getMonth(), today.getDayOfMonth(), 23, 59, 59);
return Date.from(format.atZone(ZoneId.systemDefault()).toInstant());

使用场景示例:

  • 获取本周的第一秒

private Date getThisWeekMonday() {
        LocalDate today = LocalDate.now();
        LocalDateTime format = LocalDateTime.of(today.getYear(), today.getMonth(), today.getDayOfMonth(), 0, 0, 0);
       	format= format.with(DayOfWeek.MONDAY);
        return Date.from(format.atZone(ZoneId.systemDefault()).toInstant());
    }
  • 获取本周的最后一秒
private Date getThisWeekSunday() {
        LocalDate today = LocalDate.now();
        LocalDateTime format = LocalDateTime.of(today.getYear(), today.getMonth(), today.getDayOfMonth(), 23, 59, 59);
        format=format.with(DayOfWeek.SUNDAY);
        return Date.from(format.atZone(ZoneId.systemDefault()).toInstant());
    }
  • 获取上周的第一秒
private Date getLastWeekMonday() {
        LocalDate today = LocalDate.now();
        LocalDate lastWeek = today.plusWeeks(-1);
        LocalDateTime format = LocalDateTime.of(lastWeek.getYear(), lastWeek.getMonth(), lastWeek.getDayOfMonth(), 0, 0, 0);
        format=format.with(DayOfWeek.MONDAY);
        return Date.from(format.atZone(ZoneId.systemDefault()).toInstant());
    }
  • 获取上周的最后一秒
private Date getLastWeekSunday() {
        LocalDate today = LocalDate.now();
        LocalDate lastWeek = today.plusWeeks(-1);
        LocalDateTime format = LocalDateTime.of(lastWeek.getYear(), lastWeek.getMonth(), lastWeek.getDayOfMonth(), 23, 59, 59);
        format=format.with(DayOfWeek.SUNDAY);
        return Date.from(format.atZone(ZoneId.systemDefault()).toInstant());
    }

 

使用Calendar格式化时间达不到预期的问题使用Calendar格式化时间达不到预期的问题 大大大大大碗面 发布了56 篇原创文章 · 获赞 53 · 访问量 8万+ 私信 关注
上一篇:为什么阿里规范中要求代码禁用 static 修饰 SimpleDateFormat?


下一篇:JDK 1.8 完整日期时间Api (文末附示例)