What's your time zone?
JAVA日期和時(shí)間類(lèi)徹底解決(1)
Page 1 of 3
你是否在苦苦掙扎在JAVA語(yǔ)言中的日期和時(shí)間中?當(dāng)你在計(jì)算機(jī)上顯示日期和時(shí)間時(shí),, 是否要快一個(gè)小時(shí)?或者可能要早一個(gè)小時(shí)?, 或者兩個(gè)小時(shí), 或者更嚴(yán)重? 當(dāng)你試圖用JAVA寫(xiě)日期和時(shí)間到一個(gè)文件中,或者到你的數(shù)據(jù)庫(kù)中(通過(guò)Java Database Connectivity (JDBC))— 錯(cuò)誤的時(shí)間被保存了嗎?
我曾經(jīng)被這個(gè)問(wèn)題困饒過(guò)很長(zhǎng)時(shí)間。我不能解決為什么JAVA改變了我給的時(shí)間戳(timestamps)。我從數(shù)據(jù)庫(kù)中檢索時(shí)間戳數(shù)據(jù)并顯示在我的圖形用戶(hù)接口(GUI)中時(shí), 總是會(huì)顯示一個(gè)不同的時(shí)間—和我期望的值要相差1,2或3個(gè)小時(shí)。我重新檢查了數(shù)據(jù)庫(kù)中的值,它是正常的。 那么到底應(yīng)該怎么辦呢?
調(diào)查
最終我決定對(duì)這種情況來(lái)調(diào)查一番。首先,我寫(xiě)了一個(gè)簡(jiǎn)單的JAVA類(lèi):
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 你看到了,
javap -private java.util.Calendar
在任何一種情況下,你都會(huì)發(fā)現(xiàn)在java.util.Calendar類(lèi)中有一個(gè)私有實(shí)例成員,名字為zone,它是java.util.TimeZone實(shí)例。javap結(jié)果的部分輸出顯示為:
private java.util.TimeZone zone 當(dāng)我對(duì) private transient java.util.Calendar cal; 這個(gè)就間接的表明, 然而,Java文檔告訴我們
因此你能看到,
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
類(lèi)似乎有一個(gè)類(lèi)成員是java.util.SimpleTimeZone的一個(gè)實(shí)例,
并且我也能通過(guò)一些辦法確定:
src.jar文件中可利用的源代碼
java.util.Date
類(lèi)采用了同樣的辦法時(shí),你也能發(fā)現(xiàn)它有以下成員:
Date
類(lèi)也是一個(gè)TimeZone
成員。
TimeZone
是一個(gè)抽象類(lèi),但是SimpleTimeZone
是一個(gè)聚合子類(lèi)。因此,盡管定義為成員,Calendar中的zone成員實(shí)際上是SimpleTimeZone實(shí)例,
(在J2SE 1.3中)。 這個(gè)可以通過(guò)使用上面的方法來(lái)調(diào)查T(mén)imeZone很容易得到證實(shí)。真正地,Calendar中的
是一個(gè)zone成員
SimpleTimeZone
實(shí)例。檢查DateTest
類(lèi)的輸出,它涉及到TimeZone的夏令時(shí)(DST)屬性,并為它們命名為以下屬性:
dstSavings
useDaylight
startYear
startMode
startMonth
startDay
startDayOfWeek
startTime
startTimeMode
endMode
endMonth
endDay
endDayOfWeek
endTime
endTimeMode
Date
和Calendar
類(lèi)在Daylight Saving Time上有一個(gè)概念。 檔我開(kāi)始調(diào)查這個(gè)的時(shí)候,是summer (last year), 為了調(diào)整我們服務(wù)器所有的夏令時(shí)(DST),我將系統(tǒng)時(shí)間向前挪移了一個(gè)小時(shí)。因此,我認(rèn)為JAVA將不會(huì)對(duì)DST進(jìn)行調(diào)整。