詳解Java 8時間類,越用越香
為什么會在Jdk8中加入很多時間類
非線程安全
java.util.Date 是非線程安全的,所有的日期類都是可變的,這是Java日期類最大的問題之一。
- Date date = new Date();
- for (int i = 0; i < 100; i++) {
- new Thread(new Runnable() {
- @Override
- public void run() {
- // 舉例
- int time = new Random().nextInt(100);
- date.setTime(time);
- System.out.println( Thread.currentThread().getId() + " = " + time);
- System.out.println( Thread.currentThread().getId() + " = " + date.getTime());
- }
- }).start();
- }
SimpleDateFormat格式化工具也是一樣,阿里巴巴規約中建議將SimpleDateFormat放到ThreadLocal中。
而java8中日期和時間基本都被設計final,final修飾的類,天然線程安全。
設計很差
java.util.Date同時包含日期和時間,而java.sql.Date僅包含日期,兩個類又有相同的名字,令人匪夷所思。
java.util.Date代表時間線上的一個瞬間(包含了從Unix新紀元到現在的總毫秒數),但是如果調用了Date的toString(),返回值會提示它是帶著時區的,這也會讓開發者感到疑惑。
時區、日期計算處理麻煩
日期類并不提供國際化,沒有時區支持,因此Java引入了java.util.Calendar和java.util.TimeZone類,但他們同樣存在上述所有的問題,使用復雜,不直觀
- // 獲取當前時間日歷 +8時區
- Calendar calendar = Calendar.getInstance();
- // 毫秒數
- calendar.setTimeInMillis(1601186434000L);
- // 時區轉到 utc 時間
- calendar.setTimeZone(TimeZone.getTimeZone("GMT+8:00"));
- int dstOffset = calendar.get(java.util.Calendar.DST_OFFSET);
- int zoneOffset = calendar.get(Calendar.ZONE_OFFSET);
- calendar.add(java.util.Calendar.MILLISECOND, -(zoneOffset + dstOffset));
- // 時區轉到對應的時區
- calendar.setTimeZone(TimeZone.getTimeZone("GMT+5:00"));
- int dstOffset1 = calendar.get(java.util.Calendar.DST_OFFSET);
- int zoneOffset1= calendar.get(Calendar.ZONE_OFFSET);
- calendar.add(java.util.Calendar.MILLISECOND, (zoneOffset1 + dstOffset1));
-
- // 時間計算
- calendar.add(Calendar.HOUR,15);
- // 日期計算
- calendar.add(Calendar.DAY_OF_MONTH, -1);
- // 時區計算
- calendar.add(Calendar.ZONE_OFFSET, 3);
- // 周幾
- int week = calendar.get(Calendar.DAY_OF_WEEK);
基于上述的原因,java8重新提供一套時間類,下面來看一下相關類
java8 日期、時間常見類
- ZoneId 地區 Asia/Shanghai、Europe/Paris
- ZoneOffset 偏移數據 +8:00
- Instant 它代表的是時間戳
- Duration 它表示秒或納秒時間間隔
- Period 表示一段時間的年、月、日,開使用between()方法獲取兩個日期之間的差作為Period 對象返回
- LocalDate 不包含具體時間的日期,比如2014-01-14。它可以用來存儲生日,周年紀念日,入職日期等。
- LocalTime 它代表的是不含日期的時間
- LocalDateTime 它包含了日期及時間,不過還是沒有偏移信息或者說時區。
- ZonedDateTime 這是一個包含時區的完整的日期時間,偏移量是以UTC/格林威治時間為基準的。
- OffsetDateTime 類實際上包含了LocalDateTime與ZoneOffset
- DateTimeFormatter 日期的格式化與解析,與SimpleDateFormat不同,它是不可變且線程安全的
- TemporalAdjusters 類中包含許多常用的靜態方法,避免自己編寫工具類
時間類關系圖
常見類的操作示例
- ZoneId zoneId = ZoneId.systemDefault();
- System.out.println(zoneId);//Asia/Shanghai
- ZoneOffset zoneOffset = ZoneOffset.ofHours(8);
- System.out.println(zoneOffset);//+08:00
- Instant instant = Instant.ofEpochSecond(LocalDateTime.now().toEpochSecond(ZoneOffset.ofHours(8)));
- System.out.println(instant.getEpochSecond());//1605596559
- Duration duration = Duration.between(LocalDateTime.now(), LocalDateTime.now().plusHours(1));
- System.out.println(duration.getSeconds());//3600
- Period period = Period.between(LocalDate.now(),LocalDate.now().plusDays(1));
- System.out.println(period.getDays());//1
- LocalDate date = LocalDate.now();
- System.out.println(date);//2020-11-17
- LocalTime time = LocalTime.now();
- System.out.println(time);//15:02:39.067
- LocalDateTime now = LocalDateTime.now();
- System.out.println(now);//2020-11-17T15:02:39.06
- ZonedDateTime zonedDateTime = ZonedDateTime.of(LocalDateTime.now(), zoneId);
- System.out.println(zonedDateTime);//2020-11-17T15:02:39.067+08:00[Asia/Shanghai]
- OffsetDateTime offsetDateTime = OffsetDateTime.of(LocalDateTime.now(), ZoneOffset.ofHours(8));
- System.out.println(offsetDateTime);//2020-11-17T15:02:39.068+08:00
- String format = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss").format(offsetDateTime);
- System.out.println(format);//2020-11-17 15:02:39
- TemporalAdjuster temporalAdjuster = TemporalAdjusters.firstDayOfMonth();
- System.out.println(temporalAdjuster.adjustInto(LocalDate.now()));//2020-11-01
特別說明
ZoneId、ZoneOffset主要表示時區和偏移
Instant 表示時間戳
Duration、Period 表示時間差,前者表示時間差,后者表示日期差
LocalDate、LocalTime、LocalDateTime表示日期、時間、日期+時間
ZonedDateTime、OffsetDateTime含時區信息的時間
Java8的方便之處
提供了很多時間、日期計算的方法,非常直觀

也提供了TemporalAdjusters這樣的時間工具類,內置了一些方法。
