詳解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格式化工具也是一樣,阿里巴巴規(guī)約中建議將SimpleDateFormat放到ThreadLocal中。
而java8中日期和時間基本都被設(shè)計final,final修飾的類,天然線程安全。
設(shè)計很差
java.util.Date同時包含日期和時間,而java.sql.Date僅包含日期,兩個類又有相同的名字,令人匪夷所思。
java.util.Date代表時間線上的一個瞬間(包含了從Unix新紀(jì)元到現(xiàn)在的總毫秒數(shù)),但是如果調(diào)用了Date的toString(),返回值會提示它是帶著時區(qū)的,這也會讓開發(fā)者感到疑惑。
時區(qū)、日期計算處理麻煩
日期類并不提供國際化,沒有時區(qū)支持,因此Java引入了java.util.Calendar和java.util.TimeZone類,但他們同樣存在上述所有的問題,使用復(fù)雜,不直觀
- // 獲取當(dāng)前時間日歷 +8時區(qū)
- Calendar calendar = Calendar.getInstance();
- // 毫秒數(shù)
- calendar.setTimeInMillis(1601186434000L);
- // 時區(qū)轉(zhuǎn)到 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));
- // 時區(qū)轉(zhuǎn)到對應(yīng)的時區(qū)
- 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);
- // 時區(qū)計算
- calendar.add(Calendar.ZONE_OFFSET, 3);
- // 周幾
- int week = calendar.get(Calendar.DAY_OF_WEEK);
基于上述的原因,java8重新提供一套時間類,下面來看一下相關(guān)類
java8 日期、時間常見類
- ZoneId 地區(qū) Asia/Shanghai、Europe/Paris
- ZoneOffset 偏移數(shù)據(jù) +8:00
- Instant 它代表的是時間戳
- Duration 它表示秒或納秒時間間隔
- Period 表示一段時間的年、月、日,開使用between()方法獲取兩個日期之間的差作為Period 對象返回
- LocalDate 不包含具體時間的日期,比如2014-01-14。它可以用來存儲生日,周年紀(jì)念日,入職日期等。
- LocalTime 它代表的是不含日期的時間
- LocalDateTime 它包含了日期及時間,不過還是沒有偏移信息或者說時區(qū)。
- ZonedDateTime 這是一個包含時區(qū)的完整的日期時間,偏移量是以UTC/格林威治時間為基準(zhǔn)的。
- OffsetDateTime 類實際上包含了LocalDateTime與ZoneOffset
- DateTimeFormatter 日期的格式化與解析,與SimpleDateFormat不同,它是不可變且線程安全的
- TemporalAdjusters 類中包含許多常用的靜態(tài)方法,避免自己編寫工具類
時間類關(guān)系圖
常見類的操作示例
- 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主要表示時區(qū)和偏移
Instant 表示時間戳
Duration、Period 表示時間差,前者表示時間差,后者表示日期差
LocalDate、LocalTime、LocalDateTime表示日期、時間、日期+時間
ZonedDateTime、OffsetDateTime含時區(qū)信息的時間
Java8的方便之處
提供了很多時間、日期計算的方法,非常直觀

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