1. 程式人生 > >詳解Java中的時區類TimeZone的用法

詳解Java中的時區類TimeZone的用法

void system類 深入 pri comment 相對 系統 就會 lean

一、TimeZone 簡介

TimeZone 表示時區偏移量,也可以計算夏令時。
在操作 Date, Calendar等表示日期/時間的對象時,經常會用到TimeZone;因為不同的時區,時間不同。
下面說說TimeZone對象的 2種常用創建方式。
1.獲取默認的TimeZone對象
使用方法:

TimeZone tz = TimeZone.getDefault()

2.使用 getTimeZone(String id) 方法獲取TimeZone對象
使用方法:

  1. // 獲取 “GMT+08:00”對應的時區
  2. TimeZone china = TimeZone.getTimeZone("GMT+:08:00");
  3. // 獲取 “中國/重慶”對應的時區
  4. TimeZone chongqing = TimeZone.getTimeZone("Asia/Chongqing");

關於 getTimeZone(String id) 這種方式支持的全部id參數的取值,可以通過以下方式查找:

  1. String[] ids = TimeZone.getAvailableIDs();
  2. for (String id:ids)
  3. System.out.printf(id+", ");

輸出結果:

Etc/GMT+12, Etc/GMT+11, Pacific/Midway, Pacific/Niue ....等等
例如,創建上面第2個打印值“Etc/GMT+11”對應的TimeZone。方法如下:
  1. TimeZone tz = TimeZone.getTimeZone("Etc/GMT+11");
  2. TimeZone的函數接口
  3. // 構造函數
TimeZone():
  1. Object clone()
  2. synchronized static String[] getAvailableIDs()
  3. synchronized static String[] getAvailableIDs(int offsetMillis)
  4. int getDSTSavings()
  5. synchronized static TimeZone getDefault()
  6. final String getDisplayName(Locale locale)
  7. String getDisplayName(boolean daylightTime, int style, Locale locale)
  8. final String getDisplayName()
  9. final String getDisplayName(boolean daylightTime, int style)
  10. String getID()
  11. abstract int getOffset(int era, int year, int month, int day, int dayOfWeek, int timeOfDayMillis)
  12. int getOffset(long time)
  13. abstract int getRawOffset()
  14. synchronized static TimeZone getTimeZone(String id)
  15. boolean hasSameRules(TimeZone timeZone)
  16. abstract boolean inDaylightTime(Date time)
  17. synchronized static void setDefault(TimeZone timeZone)
  18. void setID(String id)
  19. abstract void setRawOffset(int offsetMillis)
  20. abstract boolean useDaylightTime()

二、TimeZone示例:
下面通過示例演示在Date中使用TimeZone。
參考代碼如下(TimeZoneTest.java):

  1. import java.text.DateFormat;
  2. import java.util.Date;
  3. import java.util.TimeZone;
  4. /**
  5. * TimeZone的測試程序
  6. */
  7. public class TimeZoneTest {
  8. public static void main(String[] args) {
  9. // 測試創建TimeZone對象的3種方法
  10. showUsageOfTimeZones() ;
  11. // 測試TimeZone的其它API
  12. testOtherAPIs() ;
  13. // 打印getTimeZone(String id)支持的所有id
  14. //printAllTimeZones() ;
  15. }
  16. /**
  17. * 測試創建TimeZone對象的3種方法
  18. */
  19. public static void showUsageOfTimeZones() {
  20. TimeZone tz;
  21. // (01) 默認時區
  22. tz = TimeZone.getDefault();
  23. printDateIn(tz) ;
  24. // (02) 設置時區為"GMT+08:00"
  25. tz = TimeZone.getTimeZone("GMT+08:00");
  26. printDateIn(tz) ;
  27. // (03) 設置時區為""
  28. tz = TimeZone.getTimeZone("Asia/Chongqing");
  29. printDateIn(tz) ;
  30. }
  31. /**
  32. * 打印 tz對應的日期/時間
  33. */
  34. private static void printDateIn(TimeZone tz) {
  35. // date為2013-09-19 14:22:30
  36. Date date = new Date(113, 8, 19, 14, 22, 30);
  37. // 獲取默認的DateFormat,用於格式化Date
  38. DateFormat df = DateFormat.getInstance();
  39. // 設置時區為tz
  40. df.setTimeZone(tz);
  41. // 獲取格式化後的字符串
  42. String str = df.format(date);
  43. System.out.println(tz.getID()+" :"+str);
  44. }
  45. /**
  46. * 測試TimeZone的其它API
  47. */
  48. public static void testOtherAPIs() {
  49. // 默認時區
  50. TimeZone tz = TimeZone.getDefault();
  51. // 獲取“id”
  52. String id = tz.getID();
  53. // 獲取“顯示名稱”
  54. String name = tz.getDisplayName();
  55. // 獲取“時間偏移”。相對於“本初子午線”的偏移,單位是ms。
  56. int offset = tz.getRawOffset();
  57. // 獲取“時間偏移” 對應的小時
  58. int gmt = offset/(3600*1000);
  59. System.out.printf("id=%s, name=%s, offset=%s(ms), gmt=%s\n",
  60. id, name, offset, gmt);
  61. }
  62. /**
  63. * 打印getTimeZone(String id)支持的所有id
  64. */
  65. public static void printAllTimeZones() {
  66. String[] ids = TimeZone.getAvailableIDs();
  67. for (String id:ids) {
  68. //int offset = TimeZone.getTimeZone(avaIds[i]).getRawOffset();
  69. //System.out.println(i+" "+avaIds[i]+" "+offset / (3600 * 1000) + "\t");
  70. System.out.printf(id+", ");
  71. }
  72. System.out.println();
  73. }
  74. }


三、關於TimeZone和時間校準
涉及有關時間區域信息時Java和Solaris很相似。每個時間區域都有一個時間區域ID標識符。在J2SE 1.3 and 1.4中,這個ID是個字符串,是由位於J2SE 安裝程序的jre/lib子目錄中的tzmappings文件這些ID列表。 J2SE 1.3 僅僅只包含tzmappings文件,但是 J2SE 1.4包含世界不同地區的時間區域數據文件。jre/lib/zi存放著這些文件。在J2SE 1.4裏,sun.util.calendar.ZoneInfo從這些文件獲取DST規則。在Solaris中, 這些時間區域數據文件是以二進制形式存放的,不是文本文件,因此你不能看它們。 在J2SE 1.4中的時間區域數據文件和在Solaris中是不同的。

java.util.TimeZone類中getDefault方法的源代碼顯示,它最終是會調用sun.util.calendar.ZoneInfo類的getTimeZone 方法。這個方法為需要的時間區域返回一個作為ID的String參數。這個默認的時間區域ID是從 user.timezone (system)屬性那裏得到。如果user.timezone沒有定義,它就會嘗試從user.country和java.home (System)屬性來得到ID。 如果它沒有成功找到一個時間區域ID,它就會使用一個"fallback" 的GMT值。換句話說, 如果它沒有計算出你的時間區域ID,它將使用GMT作為你默認的時間區域。

註意,System屬性是在java.lang.System類的initProperties方法中被初始化的。這是一個本地方法。因此源代碼是不可用的----除非你深入到J2SE分發包中的本地代碼庫中去研究。然而,在Windows系統中,System 屬性是從Windows註冊表中被初始化的,而在Linux/Unix中是由環境變量來進行初始化。initProperties方法的Javadoc聲明,某些屬性"必須保證被定義" 且列出它們。被java.util.TimeZone類的getDefault方法使用的三個System屬性中,只有java.home作為一種“保證的”屬性在Javadoc中被列出。

推薦的解決方案 :
因此,你如何確保JAVA能給你正確的時間和日期呢?最好的辦法是確認JAVA虛擬機(JVM)的默認TimeZone類是正確的,且是適合你的地理範圍(Locale)的。你如何來確保默認TimeZone是正確的且適合的呢?這又是一個新問題了。象大多數處理的問題一樣,這個也有許多解決方案。根據java.util.TimeZone.getDefault方法的源代碼來看,最好的辦法是正確地設置user.timezone屬性。在啟動JAVA虛擬機時,你能很容易的通過使用 -D 命令 -line 參數的辦法來覆蓋(override)在java.lang.System.initProperties方法中所設置的值。例如:

java -Duser.timezone=Asia/Shanghai DateTest 

這個命令啟動DateTest類,並設置 user.timezone屬性到Asia/Shanghai。你也能夠通過使用java.lang.System 類的setProperty方法來設置user.timezone 屬性:

System.setProperty("user.timezone","Asia/Shanghai"); 

如果沒有一個可用的時間區域ID適合你,那麽就你可以創建一個自定義TimeZone 使用java.util.TimeZone 類的 setDefault 方法將它設置為默認的時間區域----就象我先前在ItsInitializer 類中所做的操作一樣。

記住,在J2SE中,大多數日期和時間相關的類都包含時間區域信息,包括那些格式類,如java.text.DateFormat, 因此它們都會被JVM的默認時間區域所影響。然而,在你創建這些類的實例時,你能為它們確保正確的時間區域信息,使得你可以更容易來設置整個JVM的默認時間區域。並且一旦設置好,就可以確保所有的這些類都將使用同一個默認的時間區域。

--------------------- 本文來自 earthchinagl 的CSDN 博客 ,全文地址請點擊:https://blog.csdn.net/earthchinagl/article/details/71404061?utm_source=copy

詳解Java中的時區類TimeZone的用法