2022-08-31  阅读(2)
原文作者:唯一浩哥 原文地址:https://www.jianshu.com/u/c52a29db48b6

一、概述

Java提供了有关时间的类和API,可以很方便的处理日期时间。

JDK 1.8之前使用的是Date和Calendar,JDK 1.8之后使用DateTime,前者毛病较多,后者更加适用。

二、基础知识

2.1 时区

我们都知道,地球是圆的,并且在不停的自转与公转,自转一圈是为一天,公转一圈是为一年。

我们这里关注自转的部分。

圆形的地球总是存在面向太阳的半球与背向太阳的半球,面向太阳的半球为白天,背向太阳的半球为黑夜,边界区域为凌晨和傍晚。

那么在这些处于地球不同位置的地区的时间也怎么分配的呢?

如果我们规定一个统一的时间,让全球共用。这样的话,0点时分,全球将拥有各种不同的景象,有的在凌晨,有的在早晨,有的在中午,有的在下午,有的在晚上等等,这也无关大雅。

但是当一个人从一个地方到另外一个地方后,他自认为的时间与光景将不再对应,这就是时差。

举个例子,比如规定中国晚上0点为统一的0点时间,而中国和美国都是在白天工作,晚上休息,但是中国是白天的时候,美国正好是晚上,一个人从中国到达美国之后,看到的就是人们在0点上班,12点休息,这与人的行为习惯相悖!

所以有了时区的概念,将全球以经线均分为24个时区,相邻时区的时间相差一个小时,当你跨越时区时,只要将你的手表调快或调慢一个小时即可,还是满足你的0点休息,12点上班的行为习惯。

实际上我们并没有绝对的使用标准的时区来划分时间,比如中国,横跨5个时区,为了国内的时间统一,我们规定统一使用北京时间为中国的时间。

2.2 格林威治时间

有了时区的概念,那么就有一个标准时间的问题了,以哪个地方为标准来定义时间呢?

那就是格林威治时间了,即GMT(Greenwich Mean Time)。

GMT表示的是格林威治平时,是指位于英国伦敦郊区的皇家格林尼治天文台当地的平太阳时,因为本初子午线被定义为通过那里的经线。

自1924年2月5日开始,格林尼治天文台负责每隔一小时向全世界发放调时信息。国际天文学联合会于1928年决定,将由格林威治平子夜起算的平太阳时作为世界时,也就是通常所说的格林威治时间。

比如中国北京位于东八区,所以北京时间即为GMT+8,即北京时间比格林威治平时快8个小时。

2.3 时间戳

时间戳是指格林威治时间1970年01月01日00时00分00秒起至现在的总秒数。

这个值是全球固定值,全球通用,虽然各地的时间不同,但是时间戳是一致的。

有了时间戳,我们就是实现不同地区时间的转换。

三、Date/Calendar/SimpleDateFormat

3.1 Date

关于Date,请查看java基础系列--Date类

3.2 Calendar

关于Calendar,请查看java基础系列--Calendar类

3.3 SimpleDateFormat

SimpleDateFormat是JDK提供的用于实现Date与String之间相互转换的工具类,成为时间格式化工具。

我们来看一下简单的格式化方式:

            public class SimpleDateFormatTest {
                public static void main(String[] args) throws ParseException {
                    test();
                }
        
                public static void test() throws ParseException {
                    SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
                    // 格式化-将Date格式化输出
                    Date date = new Date();
                    System.out.println(sdf.format(date));
                    // 解析-将String解析为Date
                    String dateStr = "2019-01-01 12:12:12";
                    System.out.println(sdf.parse(dateStr));
                }
            }

执行结果为:

            2019-02-20 18:01:30
            Tue Jan 01 12:12:12 CST 2019

最后输出了“Tue Jan 01 12:12:12 CST 2019”格式的字符串是因为使用System.out.println()方式输出自动调用了Date的toString方法。

上面的东西就是基础,重点关注的是下面的内容。

SimpleDateFormat是非线程安全的类,如果将其作为共享变量使用,在多线程环境中将会发生线程安全问题。

下面是推荐的处理办法:

  1. 作为局部变量使用
  2. 加锁
  3. 使用ThreadLocal,这也是阿里巴巴java手册中推荐的方式。
            public static ThreadLocal<SimpleDateFormat> sdfThreadLocal = new ThreadLocal<SimpleDateFormat>(){
                @Override
                protected SimpleDateFormat initialValue() {
                    return new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
                }
            };
  1. 更彻底的方式是使用DateTimeFormatter来替换。

这里只补充一点关于时区的内容:

Date其实保存的就是时间戳,我们可以通过设定时区来获取不同地点的当前时间。

            public class DateTest {
                public static void main(String[] args) {
                    Date date1 = new Date(0);// 格林威治1970年1月1日0点0分0秒
                    System.out.println("格林威治0时的时间戳:" + date1.getTime());
                    System.out.println("格林威治0时的中国时区时间:" + date1);
                    Date date2 = new Date();// 当前时间戳
                    System.out.println("date2的时间戳:" + date2.getTime());
                    System.out.println("date2表示的中国时区时间:" + date2);
                    TimeZone.setDefault(TimeZone.getTimeZone("America/Los_Angeles"));// 设置时区为美国洛杉矶时区
                    System.out.println("date2表示的美国洛杉矶时区时间:" + date2);
                    SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
                    sdf.setTimeZone(TimeZone.getTimeZone("America/New_York"));// 设置为美国纽约时区
                    System.out.println("date2表示的美国纽约时区时间:" + sdf.format(date2));
                }
            }

执行结果为:

            格林威治0时的时间戳:0
            格林威治0时的中国时区时间:Thu Jan 01 08:00:00 CST 1970
            date2的时间戳:1550656106538
            date2表示的中国时区时间:Wed Feb 20 17:48:26 CST 2019
            date2表示的美国洛杉矶时区时间:Wed Feb 20 01:48:26 PST 2019
            date2表示的美国纽约时区时间:2019-02-20 04:48:26

可见中国时区的时间确实是比格林威治时间快8个小时,而美国洛杉矶时区的时间则是比格林威治时间慢8个小时,美国纽约时区的时间又比洛杉矶快3个小时。

如果不指定时区,则自动获取操作系统的默认时区。

参考:


来源:https://www.jianshu.com/u/c52a29db48b6


Java 面试宝典是大明哥全力打造的 Java 精品面试题,它是一份靠谱、强大、详细、经典的 Java 后端面试宝典。它不仅仅只是一道道面试题,而是一套完整的 Java 知识体系,一套你 Java 知识点的扫盲贴。

它的内容包括:

  • 大厂真题:Java 面试宝典里面的题目都是最近几年的高频的大厂面试真题。
  • 原创内容:Java 面试宝典内容全部都是大明哥原创,内容全面且通俗易懂,回答部分可以直接作为面试回答内容。
  • 持续更新:一次购买,永久有效。大明哥会持续更新 3+ 年,累计更新 1000+,宝典会不断迭代更新,保证最新、最全面。
  • 覆盖全面:本宝典累计更新 1000+,从 Java 入门到 Java 架构的高频面试题,实现 360° 全覆盖。
  • 不止面试:内容包含面试题解析、内容详解、知识扩展,它不仅仅只是一份面试题,更是一套完整的 Java 知识体系。
  • 宝典详情:https://www.yuque.com/chenssy/sike-java/xvlo920axlp7sf4k
  • 宝典总览:https://www.yuque.com/chenssy/sike-java/yogsehzntzgp4ly1
  • 宝典进展:https://www.yuque.com/chenssy/sike-java/en9ned7loo47z5aw

目前 Java 面试宝典累计更新 400+ 道,总字数 42w+。大明哥还在持续更新中,下图是大明哥在 2024-12 月份的更新情况:

想了解详情的小伙伴,扫描下面二维码加大明哥微信【daming091】咨询

同时,大明哥也整理一套目前市面最常见的热点面试题。微信搜[大明哥聊 Java]或扫描下方二维码关注大明哥的原创公众号[大明哥聊 Java] ,回复【面试题】 即可免费领取。

阅读全文