欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java中的時間日期API知識點總結(jié)

 更新時間:2018年04月22日 11:11:30   投稿:laozhang  
本文給大家總結(jié)了Java中的時間日期API知識點以及相關(guān)的實例代碼分享,有興趣的朋友參考學(xué)習(xí)下。

自從 14 年發(fā)布 Java 8 以后,我們古老 java.util.Date 終于不再是我們 Java 里操作日期時間的唯一的選擇。

其實 Java 里的日期時間的相關(guān) API 一直為世猿詬病,不僅在于它設(shè)計分上工不明確,往往一個類既能處理日期又能處理時間,很混亂,還在于某些年月日期的數(shù)值映射存儲反人類,例如:0 對應(yīng)月份一月,11 對應(yīng)月份十二月,118 對應(yīng)年份 2018(1900 + 118)等。

往往我們得到某個年月值還需要再做相應(yīng)的運算才能得到準(zhǔn)確的年月日信息,直到我們的 Java 8 ,借鑒了第三方開源庫 Joda-Time 的優(yōu)秀設(shè)計,重新設(shè)計了一個日期時間 API,相比之前,可以說好用百倍,相關(guān) API 接口全部位于包 java.time 下。

古老的日期時間接口

表示時刻信息的 Date

世界上所有的計算機內(nèi)部存儲時間都使用一個 long 類型的整數(shù),而這個整數(shù)的值就是相對于英國格林尼治標(biāo)準(zhǔn)時間(1970年1月1日0時0分0秒)的毫秒數(shù)。例如:

public static void main(String[] args){
  //January 1, 1970 00:00:00 GMT.
  Date date = new Date(1000);
  System.out.println(date);
}

輸出結(jié)果:

//1970-1-1 8:00:01
Thu Jan 01 08:00:01 CST 1970

很多人可能會疑惑,1000 表示的是距離標(biāo)準(zhǔn)時間往后 1 秒,那為什么時間卻多走了 八個小時?

這和「時區(qū)」有關(guān)系,如果你位于英國的格林尼治區(qū),那么結(jié)果會如預(yù)想一樣,但是我們位于中國東八區(qū),時間要早八個小時,所以不同時區(qū)基于的基礎(chǔ)值不同。

Date 這個類以前真的扮演過很多角色,從它的源碼就可以看出來,有可以操作時刻的方法,有可以操作年月日的方法,甚至它還能管時區(qū)??梢哉f,日期時間的相關(guān)操作有它一個人就足夠了。

但這個世界就是這樣,你管的東西多了,自然就不能面面俱到,Date 中很多方法的設(shè)計并不是很合理,之前我們也說了,甚至有點反人類。所以,現(xiàn)在的 Date 類中接近百分之八十的方法都已廢棄,被標(biāo)記為 @Deprecated。

sun 公司給 Date 的目前定位是,唯一表示一個時刻,所以它的內(nèi)部應(yīng)該圍繞著那個整型的毫秒,而不再著重于各種年歷時區(qū)等信息。

Date 允許通過以下兩種構(gòu)造器實例化一個對象:

private transient long fastTime;

public Date() {
  this(System.currentTimeMillis());
}

public Date(long date) {
  fastTime = date;
}

這里的 fastTime 屬性存儲的就是時刻所對應(yīng)的毫秒數(shù),兩個構(gòu)造器還是很簡單,如果調(diào)用的是無參構(gòu)造器,那么虛擬機將以系統(tǒng)當(dāng)前的時刻值對 fastTime 進(jìn)行賦值。

還有幾個為數(shù)不多沒有被廢棄的方法:

  • public long getTime() :返回內(nèi)部存儲的毫秒數(shù)
  • public void setTime(long time):重新設(shè)置內(nèi)存的毫秒數(shù)
  • public boolean before(Date when):比較給定的時刻是否早于當(dāng)前 Date 實例
  • public boolean after(Date when):比較給定的時刻是否晚于當(dāng)前 Date 實例

還有兩個方法是 jdk1.8 以后新增的,用于向 Java 8 新增接口的轉(zhuǎn)換,待會介紹。

 描述年歷的 Calendar

Calendar 用于表示年月日等日期信息,它是一個抽象類,所以一般通過以下四種工廠方法獲取它的實例對象。

public static Calendar getInstance()

public static Calendar getInstance(TimeZone zone)

public static Calendar getInstance(Locale aLocale)

public static Calendar getInstance(TimeZone zone,Locale aLocale)

其實內(nèi)部最終會調(diào)用同一個內(nèi)部方法:

private static Calendar createCalendar(TimeZone zone,Locale aLocale)

該方法需要兩個參數(shù),一個是時區(qū),一個是國家和語言,也就是說,構(gòu)建一個 Calendar 實例最少需要提供這兩個參數(shù)信息,否則將會使用系統(tǒng)默認(rèn)的時區(qū)或語言信息。

因為不同的時區(qū)與國家語言對于時刻和年月日信息的輸出是不同的,所以這也是為什么一個 Calendar 實例必須傳入時區(qū)和國家信息的一個原因??磦€例子:

public static void main(String[] args){


  Calendar calendar = Calendar.getInstance();
  System.out.println(calendar.getTime());

  Calendar calendar1 = Calendar.getInstance
      (TimeZone.getTimeZone("GMT"), Locale.ENGLISH);
  System.out.println( calendar1.get(Calendar.YEAR) + ":" +
            calendar1.get(Calendar.HOUR) + ":" +
            calendar1.get(Calendar.MINUTE));
  }

輸出結(jié)果:

Sat Apr 21 10:32:20 CST 2018
2018:2:32

可以看到,第一個輸出為我們系統(tǒng)默認(rèn)時區(qū)與國家的當(dāng)前時間,而第二個 Calendar 實例我們指定了它位于格林尼治時區(qū)(0 時區(qū)),結(jié)果也顯而易見了,相差了八個小時,那是因為我們位于東八區(qū),時間早于 0 時區(qū)八個小時。

可能有人會疑惑了,為什么第二個 Calendar 實例的輸出要如此復(fù)雜的拼接,而不像第一個 Calendar 實例那樣直接調(diào)用 getTime 方法簡潔呢?

這涉及到 Calendar 的內(nèi)部實現(xiàn),我們一起看看:

protected long     time;

public final Date getTime() {
  return new Date(getTimeInMillis());
}

和 Date 一樣,Calendar 的內(nèi)部也維護(hù)著一個時刻信息,而 getTime 方法實際上是根據(jù)這個時刻構(gòu)建了一個 Date 對象并返回的。

而一般我們構(gòu)建 Calendar 實例的時候都不會傳入一個時刻信息,所以這個 time 的值在實例初始化的時候,程序會根據(jù)系統(tǒng)默認(rèn)的時區(qū)和當(dāng)前時間計算得到一個毫秒數(shù)并賦值給 time。

所以,所有未手動修改 time 屬性值的 Calendar 實例的內(nèi)部,time 的值都是當(dāng)時系統(tǒng)默認(rèn)時區(qū)的時刻數(shù)值。也就是說,getTime 的輸出結(jié)果是不會理會當(dāng)前實例所對應(yīng)的時區(qū)信息的,這也是我覺得 Calendar 設(shè)計的一個缺陷所在,因為這樣會導(dǎo)致兩個不同時區(qū) Calendar 實例的 getTime 輸出值只取決于實例初始化時系統(tǒng)的運行時刻。

Calendar 中也定義了很多靜態(tài)常量和一些屬性數(shù)組:

public final static int ERA = 0;

public final static int YEAR = 1;

public final static int MONTH = 2;

public final static int WEEK_OF_YEAR = 3;

public final static int WEEK_OF_MONTH = 4;

public final static int DATE = 5;
....
protected int      fields[];

protected boolean    isSet[];
...

有關(guān)日期的所有相關(guān)信息都存儲在屬性數(shù)組中,而這些靜態(tài)常量的值往往表示的就是一個索引值,通過 get 方法,我們傳入一個屬性索引,返回得到該屬性的值。例如:

Calendar myCalendar = Calendar.getInstance();
int year = myCalendar.get(Calendar.YEAR);

這里的 get 方法實際上就是直接取的 fields[1] 作為返回值,而 fields 屬性數(shù)組在 Calendar 實例初始化的時候就已經(jīng)由系統(tǒng)根據(jù)時區(qū)和語言計算并賦值了,注意,這里會根據(jù)你指定的時區(qū)進(jìn)行計算,它不像 time 始終是依照的系統(tǒng)默認(rèn)時區(qū)。

個人覺得 Calendar 的設(shè)計有優(yōu)雅的地方,也有不合理的地方,畢竟是個「古董」了,終將被替代。

DateFormat 格式化轉(zhuǎn)換

從我們之前的一個例子中可以看到,Calendar 想要輸出一個預(yù)期格式的日期信息是很麻煩的,需要自己手動拼接。而我們的 DateFormat 就是用來處理格式化字符串和日期時間之間的轉(zhuǎn)換操作的。

DateFormat 和 Calendar 一樣,也是一個抽象類,我們需要通過工廠方式產(chǎn)生其實例對象,主要有以下幾種工廠方法:

//只處理時間的轉(zhuǎn)換
public final static DateFormat getTimeInstance()

//只處理日期的轉(zhuǎn)換
public final static DateFormat getDateInstance()

//既可以處理時間,也可以處理日期
public final static DateFormat getDateTimeInstance()

當(dāng)然,它們各自都有各自的重載方法,具體的我們待會兒看。

DateFormat 有兩類方法,format 和 parse。

public final String format(Date date)

public Date parse(String source)

format 方法用于將一個日期對象格式化為字符串,parse 方法用于將一個格式化的字符串裝換為一個日期對象。例如:

public static void main(String[] args){
  Calendar calendar = Calendar.getInstance();
  DateFormat dateFormat = DateFormat.getDateTimeInstance();
  System.out.println(dateFormat.format(calendar.getTime()));
}

輸出結(jié)果:

2018-4-21 16:58:09

顯然,使用工廠構(gòu)造的 DateFormat 實例并不能夠自定義輸出格式化內(nèi)容,即輸出的字符串格式是固定的,不能滿足某些情況下的特殊需求。一般我們會直接使用它的一個實現(xiàn)類,SimpleDateFormat。

SimpleDateFormat 允許在構(gòu)造實例的時候傳入一個 pattern 參數(shù),自定義日期字符的輸出格式。例如:

public static void main(String[] args){  
  DateFormat dateFormat = new SimpleDateFormat("yyyy年MM月dd日");
  System.out.println(dateFormat.format(new Date()));
}

輸出結(jié)果:

2018年04月21日

其中,

  • yyyy:年份用四位進(jìn)行輸出
  • MM:月份用兩位進(jìn)行輸出
  • dd:兩位表示日信息
  • HH:兩位來表示小時數(shù)
  • mm:兩位表示分鐘數(shù)
  • ss:兩位來表示秒數(shù)
  • E:表示周幾,如果 Locale 在中國則會輸出 星期x,如果在美國或英國則會輸出英文的星期
  • a:表示上午或下午

當(dāng)然,對于字符串轉(zhuǎn)日期也是很方便的,允許自定義模式,但必須遵守自己制定的模式,否則程序?qū)o法成功解析。例如:

public static void main(String[] args){
  String str = "2018年4月21日 17點17分 星期六";
  DateFormat sDateFormat = new SimpleDateFormat("yyyy年M月dd日 HH點mm分 E");
  sDateFormat.parse(str);
  System.out.println(sDateFormat.getCalendar().getTime());
}

輸出結(jié)果:

Sat Apr 21 17:17:00 CST 2018

顯然,程序是正確的解析的我們的字符串并轉(zhuǎn)換為 Calendar 對象存儲在 DateFormat 內(nèi)部的。

總的來說,Date、Calendar 和 DateFormat 已經(jīng)能夠處理一般的時間日期問題了,但是不可避免的是,它們依然很繁瑣,不好用。

限于篇幅,我們下篇將對比 Java 8 的新式日期時間 API,你會發(fā)現(xiàn)它更加優(yōu)雅的設(shè)計和簡單的操作性。

相關(guān)文章

  • Mybatis foreach標(biāo)簽使用不當(dāng)導(dǎo)致異常的原因淺析

    Mybatis foreach標(biāo)簽使用不當(dāng)導(dǎo)致異常的原因淺析

    這篇文章主要介紹了Mybatis foreach標(biāo)簽使用不當(dāng)導(dǎo)致異常的原因探究,非常不錯,具有參考借鑒價值,需要的朋友可以參考下
    2016-12-12
  • Jenkins插件pipeline原理及使用方法解析

    Jenkins插件pipeline原理及使用方法解析

    這篇文章主要介紹了Jenkins插件pipeline原理及使用方法解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2020-11-11
  • Java中的訪問修飾符詳細(xì)解析

    Java中的訪問修飾符詳細(xì)解析

    以下是對Java中的訪問修飾符進(jìn)行了詳細(xì)的分析介紹,需要的朋友可以過來參考下
    2013-09-09
  • java 中String.equals和==的比較

    java 中String.equals和==的比較

    這篇文章主要介紹了java 中String.equals和==的比較的相關(guān)資料,需要的朋友可以參考下
    2017-08-08
  • Mybatis中<if>和<choose>的區(qū)別及“=”判斷方式

    Mybatis中<if>和<choose>的區(qū)別及“=”判斷方式

    這篇文章主要介紹了Mybatis中<if>和<choose>的區(qū)別及“=”判斷方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-06-06
  • HttpsURLConnection上傳文件流(實例講解)

    HttpsURLConnection上傳文件流(實例講解)

    下面小編就為大家?guī)硪黄狧ttpsURLConnection上傳文件流(實例講解)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-07-07
  • Java日常練習(xí)題,每天進(jìn)步一點點(36)

    Java日常練習(xí)題,每天進(jìn)步一點點(36)

    下面小編就為大家?guī)硪黄狫ava基礎(chǔ)的幾道練習(xí)題(分享)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧,希望可以幫到你
    2021-07-07
  • springboot優(yōu)雅獲取前端參數(shù)的方法詳解

    springboot優(yōu)雅獲取前端參數(shù)的方法詳解

    現(xiàn)在的項目基本上都是前后端分離的項目,如何打通前后端,接收前端傳過來的參數(shù)呢,這篇文章小編就來和大家詳細(xì)介紹一下springboot如何優(yōu)雅的獲取前端參數(shù)吧
    2024-03-03
  • spring中JdbcTemplate操作oracle的存儲過程實例代碼

    spring中JdbcTemplate操作oracle的存儲過程實例代碼

    JdbcTemplate是Spring對JDBC的封裝,目的是使JDBC更加易于使用,JdbcTemplate是Spring的一部分,下面這篇文章主要給大家介紹了關(guān)于spring中JdbcTemplate操作oracle的存儲過程的相關(guān)資料,需要的朋友可以參考下
    2023-04-04
  • Java并發(fā)之搞懂讀寫鎖

    Java并發(fā)之搞懂讀寫鎖

    這篇文章主要介紹了Java并發(fā)之讀寫鎖,文中相關(guān)實例代碼詳細(xì),測試可用,具有一定參考價值,需要的朋友可以了解下,希望能夠給你帶來幫助
    2021-11-11

最新評論