What's your time zone?
JAVA日期和時間類徹底解決(1)
Page 1 of 3
你是否在苦苦掙扎在JAVA語言中的日期和時間中?當你在計算機上顯示日期和時間時,, 是否要快一個小時?或者可能要早一個小時?, 或者兩個小時, 或者更嚴重? 當你試圖用JAVA寫日期和時間到一個文件中,或者到你的數據庫中(通過Java Database Connectivity (JDBC))— 錯誤的時間被保存了嗎?
我曾經被這個問題困饒過很長時間。我不能解決為什么JAVA改變了我給的時間戳(timestamps)。我從數據庫中檢索時間戳數據并顯示在我的圖形用戶接口(GUI)中時, 總是會顯示一個不同的時間—和我期望的值要相差1,2或3個小時。我重新檢查了數據庫中的值,它是正常的。 那么到底應該怎么辦呢?
調查
最終我決定對這種情況來調查一番。首先,我寫了一個簡單的JAVA類:
import java.util.*;
public class DateTest {
public static void main(String[] args) {
System.out.println("Date = " + new Date());
System.out.println("Calendar = " + Calendar.getInstance());
}
}
在Windows 98 的 Java 2 Platform, Standard Edition (J2SE) 1.3.1_01,,我得到:
Date = Tue May 06 08:13:17 IDT 2003
Calendar = java.util.GregorianCalendar[time=1052197997184,areFieldsSet=true,areAllFieldsSet
=true,lenient=false,zone=java.util.SimpleTimeZone[id=Asia/Jerusalem,offset=7200000,
dstSavings=3600000,useDaylight=true,startYear=0,startMode=1,startMonth=3,startDay=9,
startDayOfWeek=0,startTime=3600000,startTimeMode=0,endMode=1,endMonth=8,endDay=24,
endDayOfWeek=0,endTime=3600000,endTimeMode=0],firstDayOfWeek=1,minimalDaysInFirstWeek=1,
ERA=1,YEAR=2003,MONTH=4,WEEK_OF_YEAR=19,WEEK_OF_MONTH=2,DAY_OF_MONTH=6,DAY_OF_YEAR=126,
DAY_OF_WEEK=3,DAY_OF_WEEK_IN_MONTH=1,AM_PM=0,HOUR=8,HOUR_OF_DAY=8,MINUTE=13,SECOND=17,
MILLISECOND=184,ZONE_OFFSET=7200000,DST_OFFSET=3600000]
在Sun Solaris 7 with J2SE 1.3.1_02,我得到:
Date = Tue May 06 08:13:17 IDT 2003
Calendar = java.util.GregorianCalendar[time=1052197997184,areFieldsSet=true,areAllFieldsSet
=true,lenient=false,zone=java.util.SimpleTimeZone[id=Asia/Jerusalem,offset=7200000,
dstSavings=3600000,useDaylight=true,startYear=0,startMode=1,startMonth=3,startDay=9,
startDayOfWeek=0,startTime=3600000,startTimeMode=0,endMode=1,endMonth=8,endDay=24,
endDayOfWeek=0,endTime=3600000,endTimeMode=0],firstDayOfWeek=1,minimalDaysInFirstWeek=1,
ERA=1,YEAR=2003,MONTH=4,WEEK_OF_YEAR=19,WEEK_OF_MONTH=2,DAY_OF_MONTH=6,DAY_OF_YEAR=126,
DAY_OF_WEEK=3,DAY_OF_WEEK_IN_MONTH=1,AM_PM=0,HOUR=8,HOUR_OF_DAY=8,MINUTE=13,SECOND=17,
MILLISECOND=184,ZONE_OFFSET=7200000,DST_OFFSET=3600000]
在 Linux Mandrake 7.2 with J2SE 1.3.0, 我得到:
Date = Mon May 05 21:04:32 GMT+00:00 2003
Calendar = java.util.GregorianCalendar[time=1052168673155,areFieldsSet=true,areAllFieldsSet
=true,lenient=true,zone=java.util.SimpleTimeZone[id=Custom,offset=0,dstSavings=3600000,
useDaylight=false,startYear=0,startMode=0,startMonth=0,startDay=0,startDayOfWeek=0,
startTime=0,startTimeMode=0,endMode=0,endMonth=0,endDay=0,endDayOfWeek=0,endTime=0,
endTimeMode=0],firstDayOfWeek=1,minimalDaysInFirstWeek=1,ERA=1,YEAR=2003,MONTH=4,
WEEK_OF_YEAR=19,WEEK_OF_MONTH=2,DAY_OF_MONTH=5,DAY_OF_YEAR=125,DAY_OF_WEEK=2,
DAY_OF_WEEK_IN_MONTH=1,AM_PM=1,HOUR=9,HOUR_OF_DAY=21,MINUTE=4,SECOND=33,MILLISECOND=155,
ZONE_OFFSET=0,DST_OFFSET=0]
你看到了,Calendar
類似乎有一個類成員是java.util.SimpleTimeZone的一個實例,
并且我也能通過一些辦法確定:
- 使用javap實用程序,它是J2SE的一部分:
javap -private java.util.Calendar
- 檢查J2SE中
src.jar文件中可利用的源代碼
- 實用JAVA的反射機制
在任何一種情況下,你都會發現在java.util.Calendar類中有一個私有實例成員,名字為zone,它是java.util.TimeZone實例。javap結果的部分輸出顯示為:
private java.util.TimeZone zone
當我對java.util.Date
類采用了同樣的辦法時,你也能發現它有以下成員:
private transient java.util.Calendar cal;
這個就間接的表明,Date
類也是一個TimeZone
成員。
然而,Java文檔告訴我們TimeZone
是一個抽象類,但是SimpleTimeZone
是一個聚合子類。因此,盡管定義為成員,Calendar中的zone成員實際上是SimpleTimeZone實例,
(在J2SE 1.3中)。 這個可以通過使用上面的方法來調查TimeZone很容易得到證實。真正地,Calendar中的zone成員
是一個SimpleTimeZone
實例。檢查DateTest
類的輸出,它涉及到TimeZone的夏令時(DST)屬性,并為它們命名為以下屬性:
dstSavings
useDaylight
startYear
startMode
startMonth
startDay
startDayOfWeek
startTime
startTimeMode
endMode
endMonth
endDay
endDayOfWeek
endTime
endTimeMode
因此你能看到,Date
和Calendar
類在Daylight Saving Time上有一個概念。 檔我開始調查這個的時候,是summer (last year), 為了調整我們服務器所有的夏令時(DST),我將系統時間向前挪移了一個小時。因此,我認為JAVA將不會對DST進行調整。
posted on 2006-02-18 09:31
水煮三國 閱讀(916)
評論(0) 編輯 收藏