1. 程式人生 > 程式設計 >Java時區處理之Date,Calendar,TimeZone,SimpleDateFormat的區別於用法

Java時區處理之Date,Calendar,TimeZone,SimpleDateFormat的區別於用法

一、概述

1、問題描述

使用Java處理時間時,我們可能會經常發現時間不對,比如相差8個小時等等,其真實原因便是TimeZone。只有正確合理的運用TimeZone,才能保證系統時間無論何時都是準確的。由於我在外企工作,伺服器在美國,美國也有很多時區,經常會碰到向處於不同時區的伺服器發請求時需要考慮時區轉換的問題。譬如,伺服器位於西八區(GMT-8:00),而身處東八區的使用者想要查詢當天的銷售記錄。則需把東八區的“今天”這個時間範圍轉換為伺服器所在時區的時間範圍。

2、時區認識

GMT時間:即格林威治平時(Greenwich Mean Time)。平太陽時是與視太陽時對應的,由於地球軌道非圓形,執行速度隨地球與太陽距離改變而出現變化,因此視太陽時欠缺均勻性。為了糾正這種不均勻 性,天文學家就計算地球非圓形軌跡與極軸傾斜對視太陽時的效應,而平太陽時就是指經修訂之後的視太陽時。在格林威治子午線上的平太陽時稱為世界時(UTC), 又叫格林威治平時(GMT)。

3、Java 時間和時區API

3.1、Date

類Date表示特定的瞬間,精確到毫秒。獲得一個表示當前時間的Date物件有兩種方式:

Date date = new Date(); 
Date date = Calendar.getInstance().getTime(); 

Date物件本身所儲存的毫秒數可以通過date.getTime()方法得到;該函式返回自1970年1月1日 00:00:00 GMT以來此物件表示的毫秒數。它與時區和地域沒有關係(其實可以認為是GMT時間),而且還會告訴我們這個時區是否使用夏令時。有個這個資訊,我們就能夠繼續將時區物件和日期格式化器結合在一起在其它的時區和其它的語言顯示時間了。

3.2、 Calendar

Calendar的getInstance()方法有引數為TimeZone和Locale的過載,可以使用指定時區和語言環境獲得一個日曆。無參則使用預設時區和語言環境獲得日曆。

3.3、TimeZone

TimeZone物件給我們的是原始的偏移量,也就是與GMT相差的微秒數,即TimeZone表示時區偏移量,本質上以毫秒數儲存與GMT的差值。

獲取TimeZone可以通過時區ID,如"America/New_York",也可以通過GMT+/-hh:mm來設定。例如北京時間可以表示為GMT+8:00。

TimeZone.getRawOffset()方法可以用來得到當前時區的標準時間到GMT的偏移量。上段提到的"America/New_York"和"GMT+8:00"兩個時區的偏移量分別為-18000000和28800000。

4、影響TimeZone的因素

1. 作業系統的時區設定。

2. 資料傳輸時時區設定。

第一個原因其實是根本原因,當資料在不同作業系統間流轉時,就有可能因為作業系統的差異造成時間偏差,而JVM預設情況下獲取的就是作業系統的時區設定。因此在專案中最好事先設定好時區,例如:

TimeZone.setDefault(TimeZone.getTimeZone("Asia/Shanghai")); 

5、解決的方法:

從以上的分析可以看出,解決時區問題就簡單了,在時區間轉換時間時,首先用原時間減掉原時間所在時區相對於GMT的偏移量,得到原時間相對於GMT的值,然後再加上目標時區相對GMT的偏移量即可。需要注意的是這樣得到的結果依然是毫秒數,所以我們要按照指定日期格式重新轉換成Date物件即可。

6、例項:

在例項之前,假設當前的時區為中國的東八區。即GMT+8:00

package com.wsheng.aggregator.timezone;

import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.TimeZone;

/**
 * @author Josh Wang(Sheng)
 * 
 * @email [email protected]
 * 
 */
public class TimeZone1 {
	
	public static void main(String[] args) {
	  Date date = new Date(1391174450000L); // 2014-1-31 21:20:50 
	  String dateStr = "2014-1-31 21:20:50 "; 
	  SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); 
	  dateFormat.setTimeZone(TimeZone.getTimeZone("GMT")); 
	  try { 
	    Date dateTmp = dateFormat.parse(dateStr); 
	    System.out.println(dateTmp); 
	   } catch (ParseException e) { 
	    e.printStackTrace(); 
	  } 
	  String dateStrTmp = dateFormat.format(date); 
	  System.out.println(dateStrTmp); 
	}
	

}

執行結果:

Sat Feb 01 05:20:50 CST 2014
2014-01-31 13:20:50

我們發現同一時間,字串和日期執行出來的結果並不相同,那麼我們應該怎麼理解呢?

一切都要以根本原因, 即當前作業系統的時間為基準。

我的作業系統 是"Asia/Shanghai",即GMT+8的北京時間,那麼執行日期轉字串的format方法時,由於日期生成時預設是作業系統時區,因此 2014-1-31 21:20:50是北京時間,那麼推算到GMT時區,自然是要減8個小時的,即結果(2014-01-31 13:20:50);而執行字串轉日期的parse方法時,由於字串本身沒有時區的概念,因此 2013-1-31 22:17:14就是指GMT(UTC)時間【ps:所有字串都看做是GMT時間】,那麼當轉化為日期時要加上預設時區, 即"Asia/Shanghai",因此要加上8個小時。

用Calendar的話,如下:

package com.wsheng.aggregator.timezone;

import java.util.Calendar;
import java.util.Date;
import java.util.TimeZone;

/**
 * @author Josh Wang(Sheng)
 * 
 * @email [email protected]
 * 
 */
public class TimeZone2 {
	
  public static void main(String[] args) { 
  	Date date = new Date(1391174450000L); // 2014-1-31 21:20:50 
    System.out.println(date); 
    Calendar calendar = Calendar.getInstance(); 
    calendar.setTimeZone(TimeZone.getTimeZone("GMT")); 
    // 或者可以 Calendar calendar = Calendar.getInstance(TimeZone.getTimeZone("GMT")); 
    calendar.setTime(date); 
    System.out.println(calendar.get(Calendar.HOUR_OF_DAY) + ":" + calendar.get(Calendar.MINUTE)); 
  } 

}

執行結果:

Fri Jan 31 21:20:50 CST 2014
13:20

Calendar不涉及到日期與字串的轉化,因此不像SimpleDateFormat那麼複雜,與日期轉字串的思路類似。但是需要注意的是,設定完時區後,我們不能用calendar.getTime()來直接獲取Date日期,因為此時的日期與一開始setTime時是相同值,要想獲取某時區的時間,正確的做法是用calendar.get()方法,那麼我們怎麼獲得Date型別的日期呢?

正確的做法如下:

package com.wsheng.aggregator.timezone;

import java.util.Calendar;
import java.util.Date;
import java.util.TimeZone;

/**
 * @author Josh Wang(Sheng)
 * 
 * @email [email protected]
 * 
 */
public class TimeZone3 {
	
  public static void main(String[] args) { 
  	Date date = new Date(1391174450000L); // 2014-1-31 21:20:50 
    System.out.println(date); 
    Calendar calendar = Calendar.getInstance(); 
    calendar.setTimeZone(TimeZone.getTimeZone("GMT")); 
    // 或者可以 Calendar calendar = Calendar.getInstance(TimeZone.getTimeZone("GMT")); 
    calendar.setTime(date); 
    Calendar calendar2 = Calendar.getInstance(); 
    calendar2.set(calendar.get(Calendar.YEAR),calendar.get(Calendar.MONTH),calendar.get(Calendar.DAY_OF_MONTH),calendar.get(Calendar.HOUR_OF_DAY),calendar.get(Calendar.MINUTE),calendar.get(Calendar.SECOND)); 
    System.out.println(calendar2.getTime()); 
  } 

}

執行結果:

Fri Jan 31 21:20:50 CST 2014
Fri Jan 31 13:20:50 CST 2014

完美通用轉換方法

其實上面兩個轉換方法都要受到作業系統的時區設定影響,如果軟體在不同作業系統執行,仍然會有時間誤差,那麼怎麼才能統一呢?

/**
 * 
 */
package com.wsheng.aggregator.timezone;

import java.util.Date;
import java.util.TimeZone;

/**
 * @author Josh Wang(Sheng)
 * 
 * @email [email protected]
 * 
 */
public class TimeZone4 {
	
  public static void main(String[] args) { 
  	Date date = new Date(1391174450000L); // 2014-1-31 21:20:50  
    System.out.println(date); 
    date = changeTimeZone(date,TimeZone.getTimeZone("Asia/Shanghai"),TimeZone.getTimeZone("GMT")); 
    System.out.println(date); 
  } 
   
  /** 
   * 獲取更改時區後的日期 
   * @param date 日期 
   * @param oldZone 舊時區物件 
   * @param newZone 新時區物件 
   * @return 日期 
   */ 
  public static Date changeTimeZone(Date date,TimeZone oldZone,TimeZone newZone) { 
    Date dateTmp = null; 
    if (date != null) { 
      int timeOffset = oldZone.getRawOffset() - newZone.getRawOffset(); 
      dateTmp = new Date(date.getTime() - timeOffset); 
    } 
    return dateTmp; 
  } 

}

執行結果:

Fri Jan 31 21:20:50 CST 2014
Fri Jan 31 13:20:50 CST 2014

更通用的,我們可以寫一個支援型別轉換的類:

package com.wsheng.aggregator.timezone;
import java.text.*;  
import java.util.*;  
 
/**
 * 
 * @author Josh Wang(Sheng)
 * 
 * @email [email protected]
 *
 */
public class DateTransformer { 
  public static final String DATE_FORMAT = "MM/dd/yyyy HH:mm:ss"; 
     
  public static String dateTransformBetweenTimeZone(Date sourceDate,DateFormat formatter,TimeZone sourceTimeZone,TimeZone targetTimeZone) { 
    Long targetTime = sourceDate.getTime() - sourceTimeZone.getRawOffset() + targetTimeZone.getRawOffset(); 
    return DateTransformer.getTime(new Date(targetTime),formatter); 
  } 
     
  public static String getTime(Date date,DateFormat formatter){ 
    return formatter.format(date); 
  } 
     
  public static void main(String[] args){ 
    DateFormat formatter = new SimpleDateFormat(DATE_FORMAT); 
    Date date = Calendar.getInstance().getTime(); 
    System.out.println(" date: " + date);
    
    TimeZone srcTimeZone = TimeZone.getTimeZone("EST"); 
    TimeZone destTimeZone = TimeZone.getTimeZone("GMT+8"); 
    System.out.println(DateTransformer.dateTransformBetweenTimeZone(date,formatter,srcTimeZone,destTimeZone)); 
  } 
} 

DateFormat是日期/時間格式化子類的抽象類,它以與語言無關的方式格式化並解析日期或時間。日期/時間格式化子類(如 SimpleDateFormat)允許進行格式化(也就是日期 -> 文字)、解析(文字-> 日期)和標準化。將日期表示為 Date 物件,或者表示為從 GMT(格林尼治標準時間)1970 年 1 月 1 日 00:00:00 這一刻開始的毫秒數。SimpleDateFormat則是一個以與語言環境有關的方式來格式化和解析日期的具體類,可以以“日期和時間模式”字串指定日期和時間格式。我們函式中所用模式字串為"MM/dd/yyyy HH:mm:ss",則輸出日期:"07/16/2013 04:00:00"

其他常見的模式字母定義如下:

字母 日期或時間元素 表示 示例
G Era 標誌符 Text AD
y 年 Year 1996; 96
M 年中的月份 Month July; Jul; 07
w 年中的週數 Number 27
W 月份中的週數 Number 2
D 年中的天數 Number 189
d 月份中的天數 Number 10
F 月份中的星期 Number 2
E 星期中的天數 Text Tuesday; Tue
a Am/pm 標記 Text PM
H 一天中的小時數(0-23) Number 0
k 一天中的小時數(1-24) Number 24
K am/pm 中的小時數(0-11) Number 0
h am/pm 中的小時數(1-12) Number 12
m 小時中的分鐘數 Number 30
s 分鐘中的秒數 Number 55
S 毫秒數 Number 978
z 時區 General time zone Pacific Standard Time; PST; GMT-08:00
Z 時區 RFC 822 time zone -0800

由上面的分析和事例說明可知:

1. 計算機內部記錄的時間(Date date = new Date()),為格林威治標準時(GMT). 即java.util.Date代表一個時間點,其值為距公元1970年1月1日 00:00:00的毫秒數。所以它可以認為是沒有時區和Locale概念的。

2. 日期格式化類DateFormat,對於不同地區的配置一般有兩個點,一個是Locale,一個是TimeZone
前者(Locale)使DateFormat按所配置的地區特性來輸出文字(例如中國,美國,法國不同地區對日期的表示格式不一樣,中國可能是2001年10月5日)
後者(TimeZone)讓DateFormat知道怎麼去轉換,去調整時間偏移度,從而得到符合配置的時區的時間.
(即假設取得當前時間(假設當前時區為GMT+0,即與new Date()最後轉換的時間毫秒數一致)為2:00,那麼如果你配置DateFormat.setTimeZome("GMT+8"),即北京時間的時區,那麼這時候格式化輸出的就是10:00了,因為系統對原始毫秒數進行了時間偏移調整(調到你設定的時區),即加多8小時,之後再格式化輸出日期的字串形式)

3. GMT與UTC的時區是一樣的,都是以倫敦時間為基準. 而GMT+8時區就是北京時間所在時區.同一時刻的時間比GMT快8小時。

到此這篇關於Java時區處理之Date,Calendar,TimeZone,SimpleDateFormat的區別於用法的文章就介紹到這了,更多相關Java時區處理 Date,SimpleDateFormat內容請搜尋我們以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援我們!