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

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

yuyutoo 2025-04-24 10:16 13 浏览 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.前者是线程安全的,后者线程不安全。

相关推荐

ETCD 故障恢复(etc常见故障)

概述Kubernetes集群外部ETCD节点故障,导致kube-apiserver无法启动。...

在Ubuntu 16.04 LTS服务器上安装FreeRADIUS和Daloradius的方法

FreeRADIUS为AAARadiusLinux下开源解决方案,DaloRadius为图形化web管理工具。...

如何排查服务器被黑客入侵的迹象(黑客 抓取服务器数据)

---排查服务器是否被黑客入侵需要系统性地检查多个关键点,以下是一份详细的排查指南,包含具体命令、工具和应对策略:---###**一、快速初步检查**####1.**检查异常登录记录**...

使用 Fail Ban 日志分析 SSH 攻击行为

通过分析`fail2ban`日志可以识别和应对SSH暴力破解等攻击行为。以下是详细的操作流程和关键分析方法:---###**一、Fail2ban日志位置**Fail2ban的日志路径因系统配置...

《5 个实用技巧,提升你的服务器安全性,避免被黑客盯上!》

服务器的安全性至关重要,特别是在如今网络攻击频繁的情况下。如果你的服务器存在漏洞,黑客可能会利用这些漏洞进行攻击,甚至窃取数据。今天我们就来聊聊5个实用技巧,帮助你提升服务器的安全性,让你的系统更...

聊聊Spring AI Alibaba的YuQueDocumentReader

序本文主要研究一下SpringAIAlibaba的YuQueDocumentReaderYuQueDocumentReader...

Mac Docker环境,利用Canal实现MySQL同步ES

Canal的使用使用docker环境安装mysql、canal、elasticsearch,基于binlog利用canal实现mysql的数据同步到elasticsearch中,并在springboo...

RustDesk:开源远程控制工具的技术架构与全场景部署实战

一、开源远程控制领域的革新者1.1行业痛点与解决方案...

长安汽车一代CS75Plus2020款安装高德地图7.5

不用破解原车机,一代CS75Plus2020款,安装车机版高德地图7.5,有红绿灯读秒!废话不多讲,安装步骤如下:一、在拨号状态输入:在电话拨号界面,输入:*#518200#*(进入安卓设置界面,...

Zookeeper使用详解之常见操作篇(zookeeper ui)

一、Zookeeper的数据结构对于ZooKeeper而言,其存储结构类似于文件系统,也是一个树形目录服务,并通过Key-Value键值对的形式进行数据存储。其中,Key由斜线间隔的路径元素构成。对...

zk源码—4.会话的实现原理一(会话层的基本功能是什么)

大纲1.创建会话...

Zookeeper 可观测性最佳实践(zookeeper能够确保)

Zookeeper介绍ZooKeeper是一个开源的分布式协调服务,用于管理和协调分布式系统中的节点。它提供了一种高效、可靠的方式来解决分布式系统中的常见问题,如数据同步、配置管理、命名服务和集群...

服务器密码错误被锁定怎么解决(服务器密码错几次锁)

#服务器密码错误被锁定解决方案当服务器因多次密码错误导致账户被锁定时,可以按照以下步骤进行排查和解决:##一、确认锁定状态###1.检查账户锁定状态(Linux)```bash#查看账户锁定...

zk基础—4.zk实现分布式功能(分布式zk的使用)

大纲1.zk实现数据发布订阅...

《死神魂魄觉醒》卡死问题终极解决方案:从原理到实战的深度解析

在《死神魂魄觉醒》的斩魄刀交锋中,游戏卡死犹如突现的虚圈屏障,阻断玩家与尸魂界的连接。本文将从技术架构、解决方案、预防策略三个维度,深度剖析卡死问题的成因与应对之策,助力玩家突破次元壁障,畅享灵魂共鸣...

取消回复欢迎 发表评论: