Java中的時間日期處理


  程序就是輸入——>處理——>輸出。對數據的處理是程序員需要着重注意的地方,快速、高效的對數據進行處理時我們的追求。其中,時間日期的處理又尤為重要和平凡,此次,我將把Java中的時間日期處理方式進行簡單的解析,為自己以后的學習做一個備忘,也為初學者做一個借鑒。

  時間,英文Time;日期,英文Date;日歷,英文Calendar。Java中注重語義化,也是用以上的名稱對時間日期函數和相關類進行命名。

  我們將以Java自帶的時間日期類和其中的處理函數進行分析。

一、與時間日期有關的類。

  java.util.Date。實現類,其對象具有時間、日期組件。

  java.util.Calendar。抽象類,其對象具有時間、日期組件。

  java.sql.Date。實現類,其對象具有日期組件。

  java.sql.Time。實現類,其對象具有時間組件。

  java.sql.Timestamp。實現類,其對象具有時間日期組件。

  java.text.DateFormat。抽象類,其對象格式化時間日期。

  java.text.DateFormatSymbols。實現類,其對象為格式化時間日期提供參數。

  (sun.util.*canlender*.*。System。Local。TimeZone等)

  由於jdk的安裝並沒有給出全部源碼,推薦大家獲取jdk全部源碼:jdk6u23-src.rar jdk7u4-src.rar

二、類之間的關系。

  我們通過圖解和部分jdk源代碼來說明。 

  

  (上圖有幾處錯誤,Calendar拼寫錯誤。)

  以上的圖列出了部分常用的類。我們一般會使用的類java.util.Date、java.util.Calendar、java.sql.Timestamp、java.text.DateFormat進行時間日期操作,因為他們有完全的時間日期組件和全面的格式化功能。

  值得注意的是:java.sql.Date沒有時間組件!而java.sql.Time沒有日期組件!再次提醒。什么意思呢?大家請看下面的代碼:

 1 public static void main(String[] args) {
 2         /*
 3          * 以下代碼用於向大家展示各個時間日期類對象的包含組件。
 4          */
 5         java.sql.Date sqlDate = new java.sql.Date(System.currentTimeMillis());
 6         System.out.println(sqlDate.toString()); // 輸出結果:2012-09-01
 7         java.sql.Time sqlTime = new java.sql.Time(System.currentTimeMillis());
 8         System.out.println(sqlTime.toString()); // 輸出結果:12:35:11
 9         java.sql.Timestamp sqlTimestamp = new java.sql.Timestamp(System.currentTimeMillis());
10         System.out.println(sqlTimestamp.toString()); // 輸出結果:2012-09-01 12:36:33.544
11         java.util.Date utilDate = new java.util.Date(System.currentTimeMillis());
12         System.out.println(utilDate.toString()); // 輸出結果:Sat Sep 01 12:37:34 CST 2012
13         java.util.Calendar cl = java.util.Calendar.getInstance();
14         System.out.println(cl.getTime().toString()); // 輸出結果:Sat Sep 01 12:39:51 CST 2012
15     }

  可以看到:java.util.Date、java.util.Calendar、java.sql.Timestamp具有的時間日期組件(而且他們具有無參構造方法),java.sql.Date和java.sql.Time只有時間或日期組件。

  為了證實以上言論,我將部分jdk源碼貼出來供大家參考。

  java.sql.Date源代碼:

 1 package java.sql;
 2 
 3 
 4 public class Date extends java.util.Date {
 5 
 6     // 省略部分代碼……
 7 
 8     // Override all the time operations inherited from java.util.Date;
 9 
10    /**
11     * This method is deprecated and should not be used because SQL Date
12     * values do not have a time component.
13     *
14     * @deprecated
15     * @exception java.lang.IllegalArgumentException if this method is invoked
16     * @see #setHours
17     */
18     public int getHours() {
19         throw new java.lang.IllegalArgumentException();
20     }
21 
22    /**
23     * This method is deprecated and should not be used because SQL Date
24     * values do not have a time component.
25     *
26     * @deprecated
27     * @exception java.lang.IllegalArgumentException if this method is invoked
28     * @see #setMinutes
29     */
30     public int getMinutes() {
31         throw new java.lang.IllegalArgumentException();
32     }
33 
34    /**
35     * This method is deprecated and should not be used because SQL Date
36     * values do not have a time component.
37     *
38     * @deprecated
39     * @exception java.lang.IllegalArgumentException if this method is invoked
40     * @see #setSeconds
41     */
42     public int getSeconds() {
43         throw new java.lang.IllegalArgumentException();
44     }
45 
46    /**
47     * This method is deprecated and should not be used because SQL Date
48     * values do not have a time component.
49     *
50     * @deprecated
51     * @exception java.lang.IllegalArgumentException if this method is invoked
52     * @see #getHours
53     */
54     public void setHours(int i) {
55         throw new java.lang.IllegalArgumentException();
56     }
57 
58    /**
59     * This method is deprecated and should not be used because SQL Date
60     * values do not have a time component.
61     *
62     * @deprecated
63     * @exception java.lang.IllegalArgumentException if this method is invoked
64     * @see #getMinutes
65     */
66     public void setMinutes(int i) {
67         throw new java.lang.IllegalArgumentException();
68     }
69 
70    /**
71     * This method is deprecated and should not be used because SQL Date
72     * values do not have a time component.
73     *
74     * @deprecated
75     * @exception java.lang.IllegalArgumentException if this method is invoked
76     * @see #getSeconds
77     */
78     public void setSeconds(int i) {
79         throw new java.lang.IllegalArgumentException();
80     }
81 
82    /**
83     * Private serial version unique ID to ensure serialization
84     * compatibility.
85     */
86     static final long serialVersionUID = 1511598038487230103L;
87 }

  java.sql.Time源代碼:

  1     // 省略部分源代碼……
  2 
  3    /**
  4     * This method is deprecated and should not be used because SQL <code>TIME</code>
  5     * values do not have a year component.
  6     *
  7     * @deprecated
  8     * @exception java.lang.IllegalArgumentException if this
  9     *           method is invoked
 10     * @see #setYear
 11     */
 12     @Deprecated
 13     public int getYear() {
 14         throw new java.lang.IllegalArgumentException();
 15     }
 16 
 17    /**
 18     * This method is deprecated and should not be used because SQL <code>TIME</code>
 19     * values do not have a month component.
 20     *
 21     * @deprecated
 22     * @exception java.lang.IllegalArgumentException if this
 23     *           method is invoked
 24     * @see #setMonth
 25     */
 26     @Deprecated
 27     public int getMonth() {
 28         throw new java.lang.IllegalArgumentException();
 29     }
 30 
 31    /**
 32     * This method is deprecated and should not be used because SQL <code>TIME</code>
 33     * values do not have a day component.
 34     *
 35     * @deprecated
 36     * @exception java.lang.IllegalArgumentException if this
 37     *           method is invoked
 38     */
 39     @Deprecated
 40     public int getDay() {
 41         throw new java.lang.IllegalArgumentException();
 42     }
 43 
 44    /**
 45     * This method is deprecated and should not be used because SQL <code>TIME</code>
 46     * values do not have a date component.
 47     *
 48     * @deprecated
 49     * @exception java.lang.IllegalArgumentException if this
 50     *           method is invoked
 51     * @see #setDate
 52     */
 53     @Deprecated
 54     public int getDate() {
 55         throw new java.lang.IllegalArgumentException();
 56     }
 57 
 58    /**
 59     * This method is deprecated and should not be used because SQL <code>TIME</code>
 60     * values do not have a year component.
 61     *
 62     * @deprecated
 63     * @exception java.lang.IllegalArgumentException if this
 64     *           method is invoked
 65     * @see #getYear
 66     */
 67     @Deprecated
 68     public void setYear(int i) {
 69         throw new java.lang.IllegalArgumentException();
 70     }
 71 
 72    /**
 73     * This method is deprecated and should not be used because SQL <code>TIME</code>
 74     * values do not have a month component.
 75     *
 76     * @deprecated
 77     * @exception java.lang.IllegalArgumentException if this
 78     *           method is invoked
 79     * @see #getMonth
 80     */
 81     @Deprecated
 82     public void setMonth(int i) {
 83         throw new java.lang.IllegalArgumentException();
 84     }
 85 
 86    /**
 87     * This method is deprecated and should not be used because SQL <code>TIME</code>
 88     * values do not have a date component.
 89     *
 90     * @deprecated
 91     * @exception java.lang.IllegalArgumentException if this
 92     *           method is invoked
 93     * @see #getDate
 94     */
 95     @Deprecated
 96     public void setDate(int i) {
 97         throw new java.lang.IllegalArgumentException();
 98     }
 99 
100    /**
101     * Private serial version unique ID to ensure serialization
102     * compatibility.
103     */
104     static final long serialVersionUID = 8397324403548013681L;
105 }

  從上面的代碼可以看出:java.sql.Date和java.sql.Time確實是不具有完整組件的!

  我們再次利用代碼來說明:

 1 public static void main(String[] args) {
 2         /*
 3          * 以下代碼用於向大家展示各個時間日期類對象的包含組件。
 4          */
 5         java.sql.Date sqlDate = new java.sql.Date(System.currentTimeMillis());
 6         System.out.println(sqlDate.toString()); // 輸出結果:2012-09-01
 7         java.sql.Time sqlTime = new java.sql.Time(System.currentTimeMillis());
 8         System.out.println(sqlTime.toString()); // 輸出結果:12:35:11
 9         java.sql.Timestamp sqlTimestamp = new java.sql.Timestamp(System.currentTimeMillis());
10         System.out.println(sqlTimestamp.toString()); // 輸出結果:2012-09-01 12:36:33.544
11         java.util.Date utilDate = new java.util.Date(System.currentTimeMillis());
12         System.out.println(utilDate.toString()); // 輸出結果:Sat Sep 01 12:37:34 CST 2012
13         java.util.Calendar cl = java.util.Calendar.getInstance();
14         System.out.println(cl.getTime().toString()); // 輸出結果:Sat Sep 01 12:39:51 CST 2012
15         
16         /*
17          * 以下代碼用於試驗java.sql.Date和java.sql.Time是否具有完整組件。 
18          */
19         System.out.println();
20         try {
21             System.out.println(sqlDate.getHours());
22         } catch (Exception e) {
23             System.out.println(e.getMessage()); // 輸出 null
24         }
25         try {
26             System.out.println(sqlTime.getDate());
27         } catch (Exception e) {
28             System.out.println(e.getMessage()); // 輸出 null
29         }
30     }

  實驗成功,所有給大家一個忠告:在進行數據庫時間日期操作時,使用java.sql.Timestamp類。

  那么很簡單,如果您需要在程序中進行完整的時間日期操作,推薦您使用java.util.Date+java.text.DateFormat。

  如果您需要進行復雜或深入的操作,您可以選擇java.util.Calendar。有人說Calendar是Date的復雜版本,我覺得說得有一些道理。我們可以通過他們的依賴對象(通過源碼文件中引入的外部類)來證實這個說法:

  java.util.Date:

 1 package java.util;
 2 
 3 import java.text.DateFormat;
 4 import java.io.IOException;
 5 import java.io.ObjectOutputStream;
 6 import java.io.ObjectInputStream;
 7 import java.lang.ref.SoftReference;
 8 import sun.util.calendar.BaseCalendar;
 9 import sun.util.calendar.CalendarDate;
10 import sun.util.calendar.CalendarSystem;
11 import sun.util.calendar.CalendarUtils;
12 import sun.util.calendar.Era;
13 import sun.util.calendar.Gregorian;
14 import sun.util.calendar.ZoneInfo;

  java.util.Calendar:

 1 package java.util;
 2 
 3 import java.io.IOException;
 4 import java.io.ObjectInputStream;
 5 import java.io.ObjectOutputStream;
 6 import java.io.OptionalDataException;
 7 import java.io.Serializable;
 8 import java.security.AccessControlContext;
 9 import java.security.AccessController;
10 import java.security.PermissionCollection;
11 import java.security.PrivilegedActionException;
12 import java.security.PrivilegedExceptionAction;
13 import java.security.ProtectionDomain;
14 import java.text.DateFormat;
15 import java.text.DateFormatSymbols;
16 import java.util.concurrent.ConcurrentHashMap;
17 import java.util.concurrent.ConcurrentMap;
18 import sun.util.BuddhistCalendar;
19 import sun.util.calendar.ZoneInfo;
20 import sun.util.resources.LocaleData;

  java.util.Date更多地用到了sun.util.*calendar*.*。而java.util.Calendar對他們的依賴則很少,並且Calendar中加入了更好的格式化功能等……(sun.util等源碼安裝jdk不會提供,我在頂部的下載連接中提供了)。

  

  其實說這么多都是廢話。對大家有用的東西無非只有兩點:一是怎樣獲得時間日期,二是怎樣按照自定義格式顯示。

  現在我才來講解以上兩點:

    大家可以通過java.util.Date date = new java.util.Date()或者java.util.Date date = java.util.Calendar.getInstance().getTime()獲得java.util.Date對象。至少我推薦這樣做,和數據庫打交道的話就用java.sql.Timestamp。

    (而實際上jdk是不推薦我們使用java.util.Date對象來進行時間日期獲取的,我們從java.util.Date類方法注釋可以看到,基本所有的方法都有@Deprecated注解,而方法注釋大意則是"從JDK1.1開始,我們推薦您使用Calendar的靜態成員和對象成員來對時間日期進行操作"。我覺得其中的考慮可能有為了避免歧義吧,畢竟Date的意思是日期)

    大家可以通過java.text.DateFormat或者他的直接實現類java.text.SimpleDateFormat來實現時間日期的格式化。

    下面的代碼會給大家展示如何格式化時間日期:

 1 public static void main(String[] args) {
 2         /*
 3          * 以下代碼用於向大家展示各個時間日期類對象的包含組件。
 4          */
 5         java.sql.Date sqlDate = new java.sql.Date(System.currentTimeMillis());
 6         System.out.println(sqlDate.toString()); // 輸出結果:2012-09-01
 7         java.sql.Time sqlTime = new java.sql.Time(System.currentTimeMillis());
 8         System.out.println(sqlTime.toString()); // 輸出結果:12:35:11
 9         java.sql.Timestamp sqlTimestamp = new java.sql.Timestamp(System.currentTimeMillis());
10         System.out.println(sqlTimestamp.toString()); // 輸出結果:2012-09-01 12:36:33.544
11         java.util.Date utilDate = new java.util.Date(System.currentTimeMillis());
12         System.out.println(utilDate.toString()); // 輸出結果:Sat Sep 01 12:37:34 CST 2012
13         java.util.Calendar cl = java.util.Calendar.getInstance();
14         System.out.println(cl.getTime().toString()); // 輸出結果:Sat Sep 01 12:39:51 CST 2012
15         
16         /*
17          * 以下代碼用於試驗java.sql.Date和java.sql.Time是否具有完整組件。 
18          */
19         System.out.println();
20         try {
21             System.out.println(sqlDate.getHours());
22         } catch (Exception e) {
23             System.out.println(e.getMessage()); // 輸出 null
24         }
25         try {
26             System.out.println(sqlTime.getDate());
27         } catch (Exception e) {
28             System.out.println(e.getMessage()); // 輸出 null
29         }
30         
31         /*
32          * 下面的代碼給大家展示時間日期的格式化。
33          */
34         System.out.println();
35         java.text.DateFormat dateFormat = java.text.SimpleDateFormat.getInstance();
36         // java.util.Date原本的格式
37         System.out.println(utilDate.toString()); // 輸出:Sat Sep 01 13:16:13 CST 2012
38         // java.util.Date格式化后的格式
39         System.out.println(dateFormat.format(sqlDate)); // 輸出:12-9-1 下午1:16
40         System.out.println();
41         // 很多時候以上的結果並不是我們希望的,我們希望更加自由、更見簡單的操作方式
42         // 此時,java.text.SimpleDateFormat就成了我們的不二選擇
43         // SimpleDateFormat提供了無參和自定義格式參數的構造方法使我們能夠輕松地實現自定義格式化
44         java.text.SimpleDateFormat simpleDateFormat = new java.text.SimpleDateFormat("yyyy-MM-dd HH:mm:ss a");
45         System.out.println(simpleDateFormat.format(sqlDate)); // 輸出:2012-09-01 13:20:41 下午
46     }

  (我不是為了占篇幅才貼上來重復代碼的哦^_^)

  java.text.SimpleDateFormat的format方法使用參數提供了強大的格式化功能(另外,值得一提的是:它的parse方法也提供了強大的字符串轉化為Date的功能)。您可以參照以下表格進行選擇:

  

  (上圖有一出錯誤:m和mm中,前者表示當分鍾數小於10會只占用一個輸出位,即輸出0-9而不會輸出00-09)

  好了,大家趕緊利用jdk進行時間日期的操作處理吧!

 

 歡迎您移步我們的交流群,無聊的時候大家一起打發時間:Programmer Union

 或者通過QQ與我聯系:點擊這里給我發消息

 (最后編輯時間2013-04-18 10:40:56)

  


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM