joda.time之如何獲取到兩個時間的差值(正確的使用Period類)

前言

此前Java處理時間日期所使用的 Date 和 Calendar 被詬病不已,Calendar 的主要問題對象可變,而像時間和日期這樣的類應該是不可變的,另外其概念模型也有不明確的地方,月份計算從0開始等等。html

JodaTime開源時間/日期庫是很好的替代,另外Java8中也推出了新的java.time庫,設計理念與JodaTime類似。java

Joda-Time 令時間和日期值變得易於管理、操做和理解。易於使用是 Joda 的主要設計目標。Joda-Time主類 DateTime 和JDK舊有類 Date 和 Calendar之間能夠互相轉換。從而保證了與JDK框架的兼容。api

本文主要介紹了下Period類及如何正確的兩個時間的差值 經過分析構造方法的方式進行展開,詳細見下文bash

經常使用操做

org.joda.time大部分用法計算機程序的思惟邏輯使用Joda-Time優雅的處理日期時間這裏面都有很好的介紹再也不贅述,下面主要看下處理時間段(兩個日期的差值)的Period類框架

Period簡介

由一組持續時間字段值指定的不可變時間段ide

一段時間分爲多個字段如年月日來表示,這些字段由PeriodType類定義,PeriodType默認值爲standard類型,支持年,月,周,日,小時,分鐘,秒和毫秒函數

構造函數

第一種:直接聲明
public Period() {
	super(0L, (PeriodType)null, (Chronology)null);
}
第二種:傳入年、月、周、日、時、分、秒、毫秒等值 此處必定要注意中間的周這個值的定義
public Period(int var1, int var2, int var3, int var4) {
	super(0, 0, 0, 0, var1, var2, var3, var4, PeriodType.standard());
}
第三種:獲取兩個ReadableInstant實現類(如DateTime)之間的時間段
public Period(ReadableInstant var1, ReadableInstant var2, PeriodType var3) {
	super(var1, var2, var3);
}
複製代碼

正確的獲取兩個時間之間的差值

網上有不少的案例說獲取兩個時間之間的差距年月日等信息的解決方案:工具

計算兩個時間的差值post

DateTime start = new DateTime(2016,8,18,10,58);
DateTime end = new DateTime(2016,9,19,12,3);
Period period = new Period(start,end);        
System.out.println(period.getMonths()+"月"+period.getDays()+"天"
       +period.getHours()+"小時"+period.getMinutes()+"分");
複製代碼

輸出爲:ui

1月1天1小時5分
複製代碼
複製代碼

只要給定起止時間,Period就能夠自動計算出來,兩個時間之間有多少月、多少天、多少小時等。

這裏有一個問題沒有說明 拿一個案例來講

DateTime d1 = new DateTime(2018,1,27,0,30);
DateTime d2 = new DateTime(2018,2,26,2,30);
Period p0 = new Period(d1, d2);
System.out.println(p0.getYears() + "年" + p0.getMonths() + "月"  + p0.getWeeks() + "周" + p0.getDays() + "天");
複製代碼

輸出爲:

0年0月4周2天
複製代碼

是否是跟預期想要獲得的30天不一致,下面看下緣由

使用new Period(ReadableInstant var1, ReadableInstant var2)進行Period的實例化
這個構造方法默認使用的PeriodType
public Period(ReadableInstant var1, ReadableInstant var2) { 
   super(var1, var2, (PeriodType)null);
}
這時候指定的PeriodType爲null,再繼續向下看Period的上級抽象父類BasePeriod的構造方法
protected BasePeriod(ReadableInstant var1, ReadableInstant var2, PeriodType var3) {
   //1. 檢查一下var3對應的值
   var3 = this.checkPeriodType(var3);
   if (var1 == null && var2 == null) {
   	//6. 這裏將iType賦值爲var3指定的Period
   	this.iType = var3;
       this.iValues = new int[this.size()];
    } else {
    	long var4 = DateTimeUtils.getInstantMillis(var1);
       long var6 = DateTimeUtils.getInstantMillis(var2);
       Chronology var8 = DateTimeUtils.getIntervalChronology(var1, var2);
       this.iType = var3;
       this.iValues = var8.get(this, var4, var6);
    }
}
protected PeriodType checkPeriodType(PeriodType var1) {
   //2. 調用DateTimeUtils工具類的getPeriodType()傳入var1 也就是上一代碼塊的var3(也就是null)
   return DateTimeUtils.getPeriodType(var1);
}
public static final PeriodType getPeriodType(PeriodType var0) {
   //3. 在這裏就能夠看到 傳入的null最終轉化爲了PeriodType.standard()
   return var0 == null ? PeriodType.standard() : var0;
}
//4. 在PeriodType看到靜態方法 實例standard並返回對應的Period
public static PeriodType standard() {
   PeriodType var0 = cStandard;
   if (var0 == null) {
   	//5. 這裏能夠看到在Standard中指定了年月周天時分秒(有興趣的能夠看下PeriodType的實例的設計)
   	//這裏的貓膩就在於Standard中指定了周!!!
  		var0 = new PeriodType("Standard", new DurationFieldType[]{DurationFieldType.years(), DurationFieldType.months(), DurationFieldType.weeks(), DurationFieldType.days(), DurationFieldType.hours(), DurationFieldType.minutes(), DurationFieldType.seconds(), DurationFieldType.millis()}, new int[]{0, 1, 2, 3, 4, 5, 6, 7});
   	cStandard = var0;
   }

   return var0;
}
複製代碼

因此獲取兩個日期的差值如年月日的正確姿式應該以下

DateTime d1 = new DateTime(2018,1,27,0,30);
DateTime d2 = new DateTime(2018,2,26,2,30);
//指定PeriodType爲yearMonthDayTime
Period p2 = new Period(d1, d2, PeriodType.yearMonthDayTime());
System.out.println(p2.getYears() + "年" + p2.getMonths() + "月"+ p2.getWeeks() + "周" + p2.getDays() + "天");
複製代碼

輸入結果:

0年0月0周30天
複製代碼

查詢官方文檔

public Period(ReadableInstant startInstant,
              ReadableInstant endInstant)
複製代碼

Most calculations performed by this method have obvious results. The special case is where the calculation is from a "long" month to a "short" month. Here, the result favours increasing the months field rather than the days. For example, 2013-01-31 to 2013-02-28 is treated as one whole month. By contrast, 2013-01-31 to 2013-03-30 is treated as one month and 30 days (exposed as 4 weeks and 2 days). The results are explained by considering that the start date plus the calculated period result in the end date.

大體的意思是說有的兩點:

  1. 2013-01-31 to 2013-03-30 is treated as one month and 30 days (exposed as 4 weeks and 2 days)

從2013-01-31到2013-03-30被視爲有一個月和30天(其中30天暴露爲4周和2天)

上面已經分析過緣由及實現,以及若是獲取到想要的值

  1. 從大月到小月的計算須要注意,好比2013-01-31到2013-02-28是一個月

    2013-01-29到2013-02-28也是一個月

    2013-01-28到2013-02-28仍是一個月

    2013-01-27到2013-02-28是一個月零一天

    這個也是須要注意的點

總結

筆者也是在項目中使用joda.time來處理兩個時間的差值的時候使用Period這個處理時間段的類,可是在使用的過程當中出現了跟預期值不一樣的問題,隨後翻閱資料更正了用法總結了一下,但願可以幫助到你們。須要使用到更多的用法能夠參考官方文檔:User Guide

參考資料

joda.time官方API

計算機程序的思惟邏輯

相關文章
相關標籤/搜索