百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程网 > 正文

一张机,织梭光景去如飞:JDK时间变迁

yuyutoo 2025-04-24 10:16 2 浏览 0 评论

夫天地者,万物之逆旅;光阴者,百代之过客。

背景

阿里《java开发手册》中提到:

说明:如果是 JDK8 的应用,可以使用 Instant 代替 Date,LocalDateTime 代替 Calendar,
DateTimeFormatter 代替 SimpleDateFormat,官方给出的解释:simple beautiful strong immutable
thread-safe。

JDK8中的时间函数与以前的版本有个根本性的改变,完全废弃了之前关于时间函数的设计。今天就聊聊这个jdk时间的设计变迁过程。这个变迁大致分为三个过程:最初的Date设计(JDK1.0),Date设计的补丁版Calendar(JDK1.1),重新设计(JDK1.8).

Date出世

先从一个实例开始吧!

  public static void dateTest() {
    Date date=new Date();
    System.out.println(date.getYear()+" " + date.getMonth()+ " "+date.getDay());
  }

打印出的结果是:

122 6 5

注意:当前的日期是 2022-07-01

是不是很惊喜?是不是很意外?

翻看Date的说明文档,让我们来看看原因:

    /**
     * Returns a value that is the result of subtracting 1900 from the
     * year that contains or begins with the instant in time represented
     * by this <code>Date</code> object, as interpreted in the local
     * time zone.
     *
     * @return  the year represented by this date, minus 1900.
     * @see     java.util.Calendar
     * @deprecated As of JDK version 1.1,
     * replaced by <code>Calendar.get(Calendar.YEAR) - 1900</code>.
     */
    @Deprecated
    public int getYear() {
        return normalize().getYear() - 1900;
    }
    
       /**
     * Returns a number representing the month that contains or begins
     * with the instant in time represented by this <tt>Date</tt> object.
     * The value returned is between <code>0</code> and <code>11</code>,
     * with the value <code>0</code> representing January.
     *
     * @return  the month represented by this date.
     * @see     java.util.Calendar
     * @deprecated As of JDK version 1.1,
     * replaced by <code>Calendar.get(Calendar.MONTH)</code>.
     */
    @Deprecated
    public int getMonth() {
        return normalize().getMonth() - 1; // adjust 1-based to 0-based
    } 
    
    /**
     * Returns the day of the week represented by this date. The
     * returned value (<tt>0</tt> = Sunday, <tt>1</tt> = Monday,
     * <tt>2</tt> = Tuesday, <tt>3</tt> = Wednesday, <tt>4</tt> =
     * Thursday, <tt>5</tt> = Friday, <tt>6</tt> = Saturday)
     * represents the day of the week that contains or begins with
     * the instant in time represented by this <tt>Date</tt> object,
     * as interpreted in the local time zone.
     *
     * @return  the day of the week represented by this date.
     * @see     java.util.Calendar
     * @deprecated As of JDK version 1.1,
     * replaced by <code>Calendar.get(Calendar.DAY_OF_WEEK)</code>.
     */
    @Deprecated
    public int getDay() {
        return normalize().getDayOfWeek() - BaseCalendar.SUNDAY;
    }

自此,明白了:年=2022-1900=122,月=6+1=7,日=周五。

Calendar打补丁

在 1.1 版中,Calendar 类被添加到了 Java 平台中,以矫正 Date 的缺点,由此大部分的 Date 方法就都被弃用了。遗憾的是,这么做只能使情况更糟。

  public static void calendarTest() {
      Calendar cal = Calendar.getInstance();
      cal.set(2018, 12, 31); // Year, Month, Day
      System.out.print(cal.get(Calendar.YEAR) + " "+cal.get(Calendar.MONTH) + " "+cal.get(Calendar.DAY_OF_MONTH));
  }

结果显示:

2019 0 31

从上面的理解中,月份是从 0 开始的即 0~11 代表 1 月…12 月

接着 day又是从 1 开始的,为什么同一个方法设计的如此怪异?

JDK1.8:破而后立

Java 8的日期和时间类包含Instant、LocalDate、LocalTime、LocalDateTime、Duration以及Period,这些类都包含在java.time包中。

(1)Instant

在 JDK8 中,针对统计时间 等场景,推荐使用 Instant 类来代替Date。如果想获取更加精确的纳秒级时间值,使用 System.nanoTime 的方式。其使用实例如下:

  public static void instantTest() {
    Instant instant=Instant.now();
    System.out.println(instant.getEpochSecond());
    //System.out.println(System.currentTimeMillis());
    System.out.println(instant.getNano());
  }

显示结果:

1656896732
236000000

(2)使用LocalDate、LocalTime、LocalDateTime来替换Calendar

    public static void localDateTimeTest() {
    LocalDate date=LocalDate.of(2022, 7, 4);
    System.out.println(date.getYear()+" "+date.getMonthValue()+" "+date.getDayOfMonth());
    
    LocalTime time=LocalTime.of(10, 05, 30);
    System.out.println(time.getHour()+" "+time.getMinute()+" "+ time.getSecond());
    
    LocalDateTime dtime=LocalDateTime.of(2022, 7, 4, 10, 5,30);
    System.out.println(dtime.getYear()+" "+dtime.getMonthValue()+" "+dtime.getDayOfMonth()+
        " "+dtime.getHour()+" "+dtime.getMinute());
    
    }

结果显示:

2022 7 4
10 5 30
2022 7 4 10 5

(3)Duration

表示一个时间段,所以Duration类中不包含now()静态方法。可以通过Duration.between()方法创建Duration对象

    public static void durationTest() {
      LocalDateTime from = LocalDateTime.of(2022, 7, 1, 10, 7, 0);    
      LocalDateTime to = LocalDateTime.of(2022, 7, 4, 10, 7, 0);    
      Duration d = Duration.between(from, to);
      System.out.println(d.toDays()+" "+ d.toHours()+" "+d.toMinutes() +" "+d.getSeconds()+" "+ d.getNano());
    }

结果显示:

3 72 4320 259200 0

(4)Period

Period在概念上和Duration类似,区别在于Period是以年月日来衡量一个时间段,比如2年3个月6天

  public static void periodTest() {
    Period p = Period.between(
                LocalDate.of(2022, 7, 1),
                LocalDate.of(2022, 7, 4));
    System.out.println(p.getYears()+" "+ p.getMonths()+" "+p.getDays());
  }

结果显示“

0 0 3

时间转换

因历史原因,jdk1.8的时间api有时候还是需要转换为Date。

import java.time.Instant;
import java.time.LocalDate;
import java.time.LocalDateTime;
import java.time.ZoneId;
import java.util.Date;
 
public class DateUtils {
   //LocalDate 转Date 
    public static Date asDate(LocalDate localDate) {
        return Date.from(localDate.atStartOfDay().atZone(ZoneId.systemDefault()).toInstant());
    }
 //LocalDateTime 转Date 
    public static Date asDate(LocalDateTime localDateTime) {
        return Date.from(localDateTime.atZone(ZoneId.systemDefault()).toInstant());
    }
 //Date 转LocalDate 
    public static LocalDate asLocalDate(Date date) {
        return Instant.ofEpochMilli(date.getTime()).atZone(ZoneId.systemDefault()).toLocalDate();
    }
 // Date 转LocalDateTime
    public static LocalDateTime asLocalDateTime(Date date) {
        return Instant.ofEpochMilli(date.getTime()).atZone(ZoneId.systemDefault()).toLocalDateTime();
    }
}

小结

JDK1.8的时间进行重新设计,使用Instant、LocalDate、LocalTime、LocalDateTime、Duration以及Period等来替代Date、Calendar设计。使用时要注意:

(1)不要在程序中写死一年为 365 天,避免在公历闰年时出现日期转换错误或程序逻辑错误。使用LocalDate.now().lengthOfYear()获取今年的天数。

(2)不允许在程序任何地方中使用:1)java.sql.Date 2)java.sql.Time 3)java.sql.Timestamp。

(3)DateTimeFormatter 代替 SimpleDateFormat.前者是线程安全的,后者线程不安全。

相关推荐

新项目终于用上了jdk24

Java世界迎来重大更新!Oracle刚刚发布的JDK24不仅是一个长期支持版本(LTS),更是一场Java编程体验的革命。...

Ubuntu安装JDK

在Ubuntu系统上安装JDK8u441版本,可以通过多种方式实现,包括使用官方JDK的PPA仓库、下载JDK的.tar.gz文件手动安装,或者使用第三方PPA仓库如WebUpd8。以下是通过JDK...

Centos8搭建Java环境(JDK1.8+Nginx+Tomcat9+Redis+Mysql)

一、开篇1.1目的每次换新的服务器,都要找资料配下环境,所以我写这篇文章,重新梳理了一下,方便了自己,希望也能给大家带来一些帮助。安装的软件有:JDK1.8+Nginx+Tomcat9+...

测试员必备:Linux下安装JDK 1.8你必须知道的那些事

1.简介在Oracle收购Sun后,Java的一系列产品就被整合到Oracle官网中,打开官网乍眼一看也不知道去哪里下载,还得一个一个的摸索尝试,而且网上大多数都是一些Oracle收购Sun前,或者就...

真的要开始用 JDK 17 了

最近在调研JDK17,并且试着将之前的一个小项目升级了一下,在测试环境跑了一段时间。...

避坑经验分享:JDK11 与JDK1.8成功切换后,重启失效的问题

由于工作需要,本人的工作机(win10系统)安装了双JDK,分别是1.8和11。之前默认使用的是11,最近又安装了JDK1.8。从JDK11切换到JDK1.8之后发现了一个令人匪夷所思的问题,解决问题...

Center OS8安装JDK8小记

事情是这样的朋友那边是搞.NET的,要调用java那边的一个springboot的resful接口,代码写好了想调试一下,所以想把java程序部署到本地CenterOS8的虚拟机里面,折腾了半个...

JAVA新手入门必备功课~安装JDK,你学会了吗

1.Linux安装JDK下载jdk:https://www.oracle.com/technetwork/java/javase/downloads/jdk8-downloads-2133151.ht...

下载并安装JDK(Windows、Mac一篇文章搞定)

学习目的:安装JDK,并在命令行中查看当前Java版本Mac步骤:...

jenkins2.107+tomcat8+jdk1.8的安装和发布代码3种方式

jenkins2.107+tomcat8+jdk1.8的安装和发布代码3种方式如果对运维课程感兴趣,可以在b站上或csdn上搜索我的账号:运维实战课程,可以关注我,学习更多免费的运维实战技术视频...

一张机,织梭光景去如飞:JDK时间变迁

夫天地者,万物之逆旅;光阴者,百代之过客。背景阿里《java开发手册》中提到:...

JDK1.8安装&amp;环境变量配置

1、下载并安装JDK1.8链接:https://pan.baidu.com/s/1bfceFjfTQvLylu7a3T7fyg?pwd=ydtm提取码:ydtm...

宝塔面板安装jdk16 – 卸载默认的jdk1.8

昨天想安装一个halo博客,开始的时候一直安装不上,后来发现jdk版本不对,halo博客默认的jdk版本最低是jdk11,宝塔默认的是jdk1.8,所以这篇文章就来倒腾下如何在宝塔面板环境下卸载默认的...

JDK8新特性总览,官方原文说明加高质量翻译

满怀忧思,不如先干再说!通过学习,重新定义自己!前言2022年9月20号,JDK19发布,从1996年1月SUN公司发布JDK1.0正式发布已经走过了26个年头,包括三个长期支持版本JDK8、JDK1...

详解ConCurrentHashMap源码(jdk1.8)

ConCurrentHashMap是一个支持高并发集合,常用的集合之一,在jdk1.8中...

取消回复欢迎 发表评论: