Java的時(shí)間處理
計(jì)算Java日期
學(xué)習(xí)怎樣創(chuàng)建和使用日期
概要
? ? 不管你是處理財(cái)務(wù)交易還是計(jì)劃著下一步的行動(dòng),你都要知道怎樣在Java中建立,使用和顯示日期。這需要你簡(jiǎn)單的查閱一下相應(yīng)類(lèi)的API參考:一個(gè)日期可以創(chuàng)建3個(gè)相關(guān)類(lèi)的對(duì)象。這篇文章告訴你你想要知道的內(nèi)容。(3,000字)
作者:Robert Nielsen
翻譯:Cocia Lin
轉(zhuǎn)帖:蕭十一郎
? ? Java統(tǒng)計(jì)從1970年1月1日起的毫秒的數(shù)量表示日期。也就是說(shuō),例如,1970年1月2日,是在1月1日后的86,400,000毫秒。同樣的,1969年12月31日是在1970年1月1日前86,400,000毫秒。Java的Date類(lèi)使用long類(lèi)型紀(jì)錄這些毫秒值.因?yàn)閘ong是有符號(hào)整數(shù),所以日期可以在1970年1月1日之前,也可以在這之后。Long類(lèi)型表示的最大正值和最大負(fù)值可以輕松的表示290,000,000年的時(shí)間,這適合大多數(shù)人的時(shí)間要求。
Date 類(lèi)
? ?Date類(lèi)可以在java.util包中找到,用一個(gè)long類(lèi)型的值表示一個(gè)指定的時(shí)刻。它的一個(gè)有用的構(gòu)造函數(shù)是Date(),它創(chuàng)建一個(gè)表示創(chuàng)建時(shí)刻的對(duì)象。getTime()方法返回Date對(duì)象的long值。在下面的程序中,我使用Date()構(gòu)造函數(shù)創(chuàng)建一個(gè)表示程序運(yùn)行時(shí)刻的對(duì)象,并且利用getTime()方法找到這個(gè)日期代表的毫秒數(shù)量:
import java.util.*;
public class Now {
? ?public static void main(String[] args) {
? ?? ?Date now = new Date();
? ?? ?long nowLong = now.getTime();
? ?? ?System.out.println("Value is " + nowLong);
? ?}
}
當(dāng)我運(yùn)行這個(gè)程序后,我得到972,568,255,150.快速確認(rèn)一下這個(gè)數(shù)字,起碼在一個(gè)合理的范圍:它不到31年,這個(gè)數(shù)值相對(duì)1970年1月1日到我寫(xiě)這篇文章的時(shí)間來(lái)說(shuō),是合理的。計(jì)算機(jī)是這個(gè)毫秒值表示時(shí)間,人們可不愿意說(shuō)" 我將在996,321,998,34見(jiàn)到你。"幸運(yùn)的是,Java提供了一個(gè)轉(zhuǎn)換Date對(duì)象到字符串的途徑,表示成傳統(tǒng)的形式。我們?cè)谙乱还?jié)討論DateFormat類(lèi),它直觀的建立日期字符串。
DateFormat類(lèi)
DateFormat類(lèi)的一個(gè)目標(biāo)是建立一個(gè)人們能夠識(shí)別的字符串。然而,因?yàn)檎Z(yǔ)言的差別,不是所有的人希望看到嚴(yán)格的相同格式的日期。法國(guó)人更喜歡看到"25 decembre 2000,",但是美國(guó)人習(xí)慣看到"December 25,2000."所以一個(gè)DateFormat的實(shí)例創(chuàng)建以后,這個(gè)對(duì)象包含了日期的顯示格式的信息。如果使用用戶電腦區(qū)域設(shè)置缺省的格式,你可以象下面那樣,創(chuàng)建DateFormat對(duì)象,使用getDateInstance()方法:
DateFormat df = DateFormat.getDateInstance();? ?
DateFormat類(lèi)在java.text包中可以找到。
轉(zhuǎn)換成字符串
你可以使用format()方法轉(zhuǎn)換Date對(duì)象為一個(gè)字符串。下面的示例程序說(shuō)明了這個(gè)問(wèn)題:
import java.util.*;
import java.text.*;
public class NowString {
? ?public static void main(String[] args) {
? ?? ?Date now = new Date();
? ?? ?DateFormat df = DateFormat.getDateInstance();
? ?? ?String s = df.format(now);
? ?? ?System.out.println("Today is " + s);
? ?}
}??
在上面的代碼中,展示了沒(méi)有參數(shù),使用缺省格式的getDateInstance()方法。Java還提供了幾個(gè)選擇日期格式,你可以通過(guò)使用重載的getDateInstance(int style)獲得。出于方便的原因,DateFormat提供了幾種預(yù)置的常量,你可以使用這些常量參數(shù)。下面是幾個(gè)SHORT, MEDIUM, LONG, 和FULL類(lèi)型的示例:
import java.util.*;
import java.text.*;
public class StyleDemo {
? ?public static void main(String[] args) {
? ?? ?Date now = new Date();
? ?? ?DateFormat df =??DateFormat.getDateInstance();
? ?? ?DateFormat df1 = DateFormat.getDateInstance(DateFormat.SHORT);
? ?? ?DateFormat df2 = DateFormat.getDateInstance(DateFormat.MEDIUM);
? ?? ?DateFormat df3 = DateFormat.getDateInstance(DateFormat.LONG);
? ?? ?DateFormat df4 = DateFormat.getDateInstance(DateFormat.FULL);
? ?? ?String s =??df.format(now);
? ?? ?String s1 = df1.format(now);
? ?? ?String s2 = df2.format(now);
? ?? ?String s3 = df3.format(now);
? ?? ?String s4 = df4.format(now);
? ?? ?System.out.println("(Default) Today is " + s);
? ?? ?System.out.println("(SHORT)? ?Today is " + s1);
? ?? ?System.out.println("(MEDIUM)??Today is " + s2);
? ?? ?System.out.println("(LONG)? ? Today is " + s3);
? ?? ?System.out.println("(FULL)? ? Today is " + s4);
? ?}
}
程序輸出如下:
(Default) Today is Nov 8, 2000
(SHORT)? ?Today is 11/8/00
(MEDIUM)??Today is Nov 8, 2000
(LONG)? ? Today is November 8, 2000
(FULL)? ? Today is Wednesday, November 8, 2000
同樣的程序,在我的電腦上使用缺省設(shè)置運(yùn)行后,改變區(qū)域設(shè)置為瑞典,輸出如下:
(Default) Today is 2000-nov-08
(SHORT)? ?Today is 2000-11-08
(MEDIUM)??Today is 2000-nov-08
(LONG)? ? Today is den 8 november 2000
(FULL)? ? Today is den 8 november 2000? ???
??
從這里,你能看到,瑞典的月份不是大寫(xiě)的(雖然November還是november).還有,LONG和FULL版本在瑞典語(yǔ)中是一樣的,但是美國(guó)英語(yǔ)卻不同。另外,有趣的是,瑞典語(yǔ)單詞的星期三,onsdag,沒(méi)有包含在FULL日期里,英語(yǔ)卻包括。
注意你能夠使用getDateInstance()方法改變DateFormat實(shí)例的語(yǔ)種;但是,在上面的例子中,是通過(guò)改變Windows98的控制面板的區(qū)域設(shè)置做到的。不同的地方的區(qū)域設(shè)置不同,結(jié)果就不同,這樣有好處,也有不足,Java程序員應(yīng)該了解這些。一個(gè)好處是Java程序員可以只寫(xiě)一行代碼就可以顯示日期,而且世界不同地區(qū)的電腦運(yùn)行同樣的程序會(huì)有不用的日期格式。 但是這也是一個(gè)缺點(diǎn),當(dāng)程序員希望顯示同一種格式的時(shí)--這也有可取之處,舉例來(lái)說(shuō),在程序中混合輸出文本和日期,如果文本是英文,我們就不希望日期格式是其他的格式,象德文或是西班牙文。如果程序員依靠日期格式編程,日期格式將根據(jù)運(yùn)行程序所在電腦的區(qū)域設(shè)置不用而不同。
解析字符串
通過(guò)parse()方法,DateFormat能夠以一個(gè)字符串創(chuàng)立一個(gè)Date對(duì)象。這個(gè)方法能拋出ParseException異常,所以你必須使用適當(dāng)?shù)漠惓L幚砑夹g(shù)。下面的例子程序通過(guò)字符串創(chuàng)建Date對(duì)象:
import java.util.*;
import java.text.*;
public class ParseExample {
? ?public static void main(String[] args) {
? ?? ?String ds = "November 1, 2000";
? ?? ?DateFormat df = DateFormat.getDateInstance();
? ?? ?try {
? ?? ?? ?Date d = df.parse(ds);
? ?? ?}
? ?? ?catch(ParseException e) {
? ?? ?? ?System.out.println("Unable to parse " + ds);
? ?? ?}
? ?}
}
在創(chuàng)建一個(gè)任意的日期時(shí)parse()方法很有用。我將通過(guò)另一種方法創(chuàng)建一個(gè)任意得日期。同時(shí),你將看到怎樣進(jìn)行基本日期計(jì)算,例如計(jì)算90天后的另一天。你可以使用GregorianCalendar類(lèi)來(lái)完成這個(gè)任務(wù)。
GregorianCalendar類(lèi)
創(chuàng)建一個(gè)代表任意日期的一個(gè)途徑使用GregorianCalendar類(lèi)的構(gòu)造函數(shù),它包含在java.util包中:
GregorianCalendar(int year, int month, int date)
注意月份的表示,一月是0,二月是1,以此類(lèi)推,是12月是11。因?yàn)榇蠖鄶?shù)人習(xí)慣于使用單詞而不是使用數(shù)字來(lái)表示月份,這樣程序也許更易讀,父類(lèi)Calendar使用常量來(lái)表示月份:JANUARY, FEBRUARY,等等。所以,創(chuàng)建Wilbur 和 Orville制造第一架動(dòng)力飛機(jī)的日期(December 17, 1903),你可以使用:
GregorianCalendar firstFlight = new GregorianCalendar(1903, Calendar.DECEMBER, 17);??
出于清楚的考慮,你應(yīng)該使用前面的形式。但是,你也應(yīng)該學(xué)習(xí)怎樣閱讀下面的短格式。下面的例子同樣表示December 17,1903(記住,在短格式中,11表示December)
GregorianCalendar firstFlight = new GregorianCalendar(1903, 11, 17);? ?
在上一節(jié)中,你學(xué)習(xí)了轉(zhuǎn)換Date對(duì)象到字符串。這里,你可以做同樣的事情;但是首先,你需要將GregorianCalendar對(duì)象轉(zhuǎn)換到Date。要做到這一點(diǎn),你可以使用getTime()方法,從它得父類(lèi)Calendar繼承而來(lái)。GetTime()方法返回GregorianCalendar相應(yīng)的Date對(duì)象。你能夠創(chuàng)建GregorianCalendar對(duì)象,轉(zhuǎn)換到Date對(duì)象,得到和輸出相應(yīng)的字符串這樣一個(gè)過(guò)程。下面是例子:
import java.util.*;
import java.text.*;
public class Flight {
? ?public static void main(String[] args) {
? ?? ?GregorianCalendar firstFlight = new GregorianCalendar(1903, Calendar.DECEMBER, 17);? ?
? ?? ?Date d = firstFlight.getTime();
? ?? ?DateFormat df = DateFormat.getDateInstance();
? ?? ?String s = df.format(d);
? ?? ?System.out.println("First flight was " + s);
? ?}
}
有時(shí)候創(chuàng)建一個(gè)代表當(dāng)前時(shí)刻的GregorianCalendar類(lèi)的實(shí)例是很有用的。你可以簡(jiǎn)單的使用沒(méi)有參數(shù)的GregorianCalendar構(gòu)造函數(shù),象這樣:
GregorianCalendar thisday = new GregorianCalendar();
一個(gè)輸出今天日期的例子程序,使用GregorianCalendar對(duì)象:
import java.util.*;
import java.text.*;
class Today {
? ?public static void main(String[] args) {
? ?? ?GregorianCalendar thisday = new GregorianCalendar();
? ?? ?Date d = thisday.getTime();
? ?? ?DateFormat df = DateFormat.getDateInstance();
? ?? ?String s = df.format(d);
? ?? ?System.out.println("Today is " + s);
? ?}
}
注意到,Date()構(gòu)造函數(shù)和GregorianCalendar()構(gòu)造函數(shù)很類(lèi)似:都創(chuàng)建一個(gè)對(duì)象,條件簡(jiǎn)單,代表今天。
日期處理
GregorianCalendar類(lèi)提供處理日期的方法。一個(gè)有用的方法是add().使用add()方法,你能夠增加象年,月數(shù),天數(shù)到日期對(duì)象中
R褂胊dd()方法,你必須提供要增加的字段,要增加的數(shù)量。一些有用的字段是DATE, MONTH, YEAR, 和 WEEK_OF_YEAR。下面的程序使用add()方法計(jì)算未來(lái)80天的一個(gè)日期。在Jules的<環(huán)球80天>是一個(gè)重要的數(shù)字,使用這個(gè)程序可以計(jì)算Phileas Fogg從出發(fā)的那一天1872年10月2日后80天的日期:
import java.util.*;
import java.text.*;
public class World {
? ?public static void main(String[] args) {
? ?? ?GregorianCalendar worldTour = new GregorianCalendar(1872, Calendar.OCTOBER, 2);
? ?? ?worldTour.add(GregorianCalendar.DATE, 80);
? ?? ?Date d = worldTour.getTime();
? ?? ?DateFormat df = DateFormat.getDateInstance();
? ?? ?String s = df.format(d);
? ?? ?System.out.println("80 day trip will end " + s);
? ?}
}
這個(gè)例子是想象的,但在一個(gè)日期上增加天數(shù)是一個(gè)普遍的操作:影碟可以租3天,圖書(shū)館可以借書(shū)21天,商店經(jīng)常需要將購(gòu)買(mǎi)的物品在30天內(nèi)賣(mài)出。下面的程序演示了使用年計(jì)算:
import java.util.*;
import java.text.*;
public class Mortgage {
? ?public static void main(String[] args) {
? ?? ?GregorianCalendar mortgage = new GregorianCalendar(1997, Calendar.MAY, 18);
? ?? ?mortgage.add(Calendar.YEAR, 15);
? ?? ?Date d = mortgage.getTime();
? ?? ?DateFormat df = DateFormat.getDateInstance();
? ?? ?String s = df.format(d);
? ?? ?System.out.println("15 year mortgage amortized on " + s);? ? }
}
? ? add()一個(gè)重要的副作用是它改變的原來(lái)的日期。有時(shí)候,擁有原始日期和修改后的日期很重要。不幸的是,你不能簡(jiǎn)單的創(chuàng)建一個(gè)GregorianCalendar對(duì)象,設(shè)置它和原來(lái)的相等(equal)。原因是兩個(gè)變量指向同一個(gè)Date()對(duì)象地址。如果Date對(duì)象改變,兩個(gè)變量就指向改變后的日期對(duì)象。代替這種做法,應(yīng)該創(chuàng)建一個(gè)新對(duì)象。下面的程序示范了這種做法:
import java.util.*;
import java.text.*;
public class ThreeDates {
? ?public static void main(String[] args) {
? ?? ?GregorianCalendar gc1 = new GregorianCalendar(2000, Calendar.JANUARY, 1);
? ?? ?GregorianCalendar gc2 = gc1;
? ?? ?GregorianCalendar gc3 = new GregorianCalendar(2000, Calendar.JANUARY, 1);
? ?? ?//Three dates all equal to January 1, 2000
? ?? ?gc1.add(Calendar.YEAR, 1);
? ?? ?file://gc1 and gc2 are changed
? ?? ?DateFormat df = DateFormat.getDateInstance();
? ?? ?Date d1 = gc1.getTime();
? ?? ?Date d2 = gc2.getTime();
? ?? ?Date d3 = gc3.getTime();
? ?? ?String s1 = df.format(d1);
? ?? ?String s2 = df.format(d2);
? ?? ?String s3 = df.format(d3);
? ?? ?System.out.println("gc1 is " + s1);
? ?? ?System.out.println("gc2 is " + s2);
? ?? ?System.out.println("gc3 is " + s3);
? ?}
}
? ? 程序運(yùn)行后,gc1和gc2被變成2001年(因?yàn)閮蓚€(gè)對(duì)象指向同一個(gè)Date,而Date已經(jīng)被改變了)。對(duì)象gc3指向一個(gè)單獨(dú)的Date,它沒(méi)有被改變。
計(jì)算復(fù)習(xí)日期
在這節(jié),你將看到一個(gè)依據(jù)現(xiàn)實(shí)世界的例子。這個(gè)詳細(xì)的程序計(jì)算過(guò)去一個(gè)具體的日期。例如,你閱讀這篇文章,你想要記住一個(gè)印象深刻的知識(shí)點(diǎn)。如果你沒(méi)有照片一樣的記憶力,你就要定期的復(fù)習(xí)這些新資料,這將幫助你記住它。關(guān)于復(fù)習(xí)系統(tǒng),Kurt Hanks 和 Gerreld L. Pulsipher在他們的< Five Secrets to Personal Productivity個(gè)人能力的5個(gè)秘密>中有討論,建議看過(guò)第一眼后馬上回顧一下,然后是1天后,1個(gè)星期后,1個(gè)月后,3個(gè)月后,1年后。我的這篇文章,你要馬上回顧一下,從現(xiàn)在算起,再就是明天,然后是1個(gè)星期,1個(gè)月,3個(gè)月,1年后。我們的程序?qū)⒂?jì)算這些日期。
這個(gè)程序非常有用的,它將是PIM(Personal Information Manager個(gè)人信息管理器)的一個(gè)組成部分,并將確定復(fù)習(xí)時(shí)間。在下面的程序中,getDates()方法對(duì)一個(gè)返回日期數(shù)組(復(fù)習(xí)日期)的電子軟件很有用。另外,你可以返回單獨(dú)的一個(gè)日期,使用getFirstDay(),getOneDay(),getOneWeek(),getOnMonth()和getOneYear().當(dāng)時(shí)間范圍超出這個(gè)PIM的ReviewDates的計(jì)算范圍時(shí)ReviewDates類(lèi)演示了怎樣計(jì)算時(shí)間段。現(xiàn)在,你可以容易的修改它用來(lái)處理你需要的時(shí)間段,象圖書(shū)館借書(shū),錄影帶租賃和抵押計(jì)算。首先,ReviewDates類(lèi)顯示在下面:
import java.util.*;
import java.text.*;
public class ReviewDates {
? ?private GregorianCalendar firstDay, oneDay, oneWeek, oneMonth, oneQuarter, oneYear;
? ?final int dateArraySize = 6;
? ?ReviewDates(GregorianCalendar gcDate) {
? ?? ?int year = gcDate.get(GregorianCalendar.YEAR);
? ?? ?int month = gcDate.get(GregorianCalendar.MONTH);
? ?? ?int date = gcDate.get(GregorianCalendar.DATE);
? ?? ?firstDay = new GregorianCalendar(year, month, date);
? ?? ?oneDay = new GregorianCalendar(year, month, date);
? ?? ?oneWeek = new GregorianCalendar(year, month, date);
? ?? ?oneMonth = new GregorianCalendar(year, month, date);
? ?? ?oneQuarter = new GregorianCalendar(year, month, date);
? ?? ?oneYear = new GregorianCalendar(year, month, date);
? ?? ?oneDay.add(GregorianCalendar.DATE, 1);
? ?? ?oneWeek.add(GregorianCalendar.DATE, 7);
? ?? ?oneMonth.add(GregorianCalendar.MONTH, 1);
? ?? ?oneQuarter.add(GregorianCalendar.MONTH, 3);
? ?? ?oneYear.add(GregorianCalendar.YEAR, 1);
? ?}
? ?ReviewDates() {
? ?? ?this(new GregorianCalendar());
? ?}
? ?public void listDates() {
? ?? ?DateFormat df = DateFormat.getDateInstance(DateFormat.LONG);
? ?? ?Date startDate = firstDay.getTime();
? ?? ?Date date1 = oneDay.getTime();
? ?? ?Date date2 = oneWeek.getTime();
? ?? ?Date date3 = oneMonth.getTime();
? ?? ?Date date4 = oneQuarter.getTime();
? ?? ?Date date5 = oneYear.getTime();
? ?? ?String ss =??df.format(startDate);
? ?? ?String ss1 = df.format(date1);
? ?? ?String ss2 = df.format(date2);
? ?? ?String ss3 = df.format(date3);
? ?? ?String ss4 = df.format(date4);
? ?? ?String ss5 = df.format(date5);
? ?? ?System.out.println("Start date is " + ss);
? ?? ?System.out.println("Following review dates are:");
? ?? ?System.out.println(ss1);
? ?? ?System.out.println(ss2);
? ?? ?System.out.println(ss3);
? ?? ?System.out.println(ss4);
? ?? ?System.out.println(ss5);
? ?? ?System.out.println();
? ?}
? ?public GregorianCalendar[] getDates() {
? ?? ?GregorianCalendar[] memoryDates = new GregorianCalendar[dateArraySize];
? ?? ?memoryDates[0] = firstDay;
? ?? ?memoryDates[1] = oneDay;
? ?? ?memoryDates[2] = oneWeek;
? ?? ?memoryDates[3] = oneMonth;
? ?? ?memoryDates[4] = oneQuarter;
? ?? ?memoryDates[5] = oneYear;
? ?? ?return memoryDates;
? ?}
? ?public GregorianCalendar getFirstDay() {
? ?? ?return this.firstDay;
? ?}
? ?public GregorianCalendar getOneDay() {
? ?? ?return this.oneDay;
? ?}
? ?public GregorianCalendar getOneWeek() {
? ?? ?return this.oneWeek;
? ?}
? ?public GregorianCalendar getOneMonth() {
? ?? ?return this.oneMonth;
? ?}
? ?public GregorianCalendar getOneQuarter() {
? ?? ?return this.oneQuarter;
? ?}
? ?public GregorianCalendar getOneYear() {
? ?? ?return this.oneYear;
? ?}
}
下面是使用ReviewDates類(lèi)列出復(fù)習(xí)日期的例子程序:
import java.util.*;
public class ShowDates {
? ?public static void main(String[] args) {
? ?? ?ReviewDates rd = new ReviewDates();
? ?? ?rd.listDates();
? ?? ?GregorianCalendar gc = new GregorianCalendar(2001, Calendar.JANUARY, 15);
? ?? ?ReviewDates jan15 = new ReviewDates(gc);
? ?? ?jan15.listDates();
? ?}
}
總結(jié)
這篇文章介紹了關(guān)于日期處理的3個(gè)重要的類(lèi):Date,DateFormat,GregorianCalendar.這些類(lèi)讓你創(chuàng)建日期,轉(zhuǎn)換成字符串,和計(jì)算日期基本元素。處理Java中的日期問(wèn)題,這篇文章只是冰山一角。可是,我在這里介紹的類(lèi)和方法不僅僅是你學(xué)習(xí)高級(jí)技術(shù)的跳板,這些類(lèi)和方法本身就可以處理很多通常的日期相關(guān)的任務(wù)
關(guān)于作者
Robert Nielsen是SCJP。他擁有碩士學(xué)位,專(zhuān)攻計(jì)算機(jī)教育,并且在計(jì)算機(jī)領(lǐng)域執(zhí)教多年
1. Java計(jì)算時(shí)間依靠1970年1月1日開(kāi)始的毫秒數(shù).??????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????
2. Date類(lèi)的構(gòu)造函數(shù)Date()返回代表當(dāng)前創(chuàng)建的時(shí)刻的對(duì)象。Date的方法getTime()返回一個(gè)long值在數(shù)值上等于1970年1月1日之前或之后的時(shí)刻。
3. DateFormat類(lèi)用來(lái)轉(zhuǎn)換Date到String,反之亦然。靜態(tài)方法getDateInstance()返回DateFormat的缺省格式;getDateInstance(DateFormat.FIELD)返回指定的DateFormat對(duì)象格式。Format(Date d)方法返回String表示日期,例如"January 1,2002."反過(guò)來(lái),parse(String s)方法返回以參數(shù)字符串表示的Date對(duì)象。
4. format()方法返回的字符串格式根據(jù)不同地區(qū)的時(shí)間設(shè)置而有所不同。
5. GregorianCalendear類(lèi)有兩個(gè)重要的構(gòu)造函數(shù):GregorianCalerdar(),返回代表當(dāng)前創(chuàng)建時(shí)間的對(duì)象;GregorianCalendar(int year,int month,int date)返回代表任意日期的對(duì)象。GregorianCalendar類(lèi)的getTime()方法返回日期對(duì)象。Add(int field,int amount)方法通過(guò)加或減時(shí)間單位,象天數(shù),月數(shù)或年數(shù)來(lái)計(jì)算日期。
GregorianCalendar和 時(shí)間
?兩個(gè)GregorianCalendar的構(gòu)造函數(shù)可以用來(lái)處理時(shí)間。前者創(chuàng)建一個(gè)表示日期,小時(shí)和分鐘的對(duì)象:
GregorianCalendar(int year, int month, int date, int hour, int minute)
第二個(gè)創(chuàng)建一個(gè)表示一個(gè)日期,小時(shí),分鐘和秒:
GregorianCalendar(int year, int month, int date, int hour, int minute, int second)
首先,我應(yīng)該提醒一下,每一個(gè)構(gòu)造函數(shù)需要時(shí)間信息中的日期信息(年,月,日)。如果你想說(shuō)2:30 p.m.,你必須指出日期。
同樣,每一個(gè)GregorianCalendar構(gòu)造函數(shù)創(chuàng)建一個(gè)在時(shí)間上使用毫秒計(jì)算的對(duì)象。所以,如果你的構(gòu)造函數(shù)只提供年,月,日參數(shù),那小時(shí),分鐘,秒和毫秒的值將被置0.
DateFormat和時(shí)間
你可以使用靜態(tài)方法getDateTimeInstance(int dateStyle,int timeStyle)來(lái)建立DateFormat對(duì)象來(lái)顯示時(shí)間和日期。這個(gè)方法表明你想要的日期和時(shí)間格式。如果你喜歡使用缺省格式,可以使用getDateTimeInstance()來(lái)代替它。
你可以使用靜態(tài)方法getTimeInstance(int timeStyle)創(chuàng)建DateFormat對(duì)象來(lái)顯示正確的時(shí)間。
下面的程序示范了getDateTimeInstance()和getTimeInstance()怎樣工作:
import java.util.*;
import java.text.*;
public class Apollo {
?? public static void main(String[] args) {
????? GregorianCalendar liftOffApollo11 = new GregorianCalendar(1969, Calendar.JULY, 16, 9, 32);
????? Date d = liftOffApollo11.getTime();
????? DateFormat df1 = DateFormat.getDateTimeInstance(DateFormat.MEDIUM, DateFormat.MEDIUM);
????? DateFormat df2 = DateFormat.getTimeInstance(DateFormat.SHORT);
????? String s1 = df1.format(d);
????? String s2 = df2.format(d);
????? System.out.println(s1);
????? System.out.println(s2);
?? }
}??
???????
在我的電腦上,上面的程序顯示如下:
Jul 16, 1969 9:32:00 AM
9:32 AM
(輸出根據(jù)你所在得地區(qū)有所不同)
計(jì)算時(shí)間間隔
???? 你可能有時(shí)需要計(jì)算過(guò)去的時(shí)間;例如,給你開(kāi)始和結(jié)束時(shí)間,你想知道制造流程的持續(xù)時(shí)間。一個(gè)出租公司按小時(shí)或天數(shù)出租東西,計(jì)算時(shí)間對(duì)他們也很有用。同樣的,在金融界,經(jīng)常需要計(jì)算重要的支付時(shí)間。
將問(wèn)題復(fù)雜化,人類(lèi)至少是用兩種方法計(jì)算時(shí)間。你可以說(shuō)一天已經(jīng)結(jié)束當(dāng)24小時(shí)過(guò)去了,或者日歷從今天翻到明天。我們將討論我們想到的這兩種情況。
時(shí)間段,情況 1:嚴(yán)格時(shí)間單位
在這種情況中,只有24小時(shí)過(guò)去,這天才過(guò)去,60分鐘過(guò)去,這個(gè)小時(shí)才過(guò)去,60秒過(guò)去,這個(gè)分鐘才過(guò)去,以此類(lèi)推。在這個(gè)方法中,23小時(shí)的時(shí)間將被認(rèn)為是0天。
使用這種方法計(jì)算時(shí)間段,你從計(jì)算過(guò)去的毫秒開(kāi)始。為了做到這一點(diǎn),首先轉(zhuǎn)換每個(gè)日期為從1970年1月1日起得毫秒數(shù)。你可以從第二個(gè)毫秒值中減去第一個(gè)毫秒值。這里有一個(gè)簡(jiǎn)單的計(jì)算:
import java.util.*;
public class ElapsedMillis {
?? public static void main(String[] args) {
????? GregorianCalendar gc1 = new GregorianCalendar(1995, 11, 1, 3, 2, 1);
????? GregorianCalendar gc2 = new GregorianCalendar(1995, 11, 1, 3, 2, 2);
????? // the above two dates are one second apart
????? Date d1 = gc1.getTime();
????? Date d2 = gc2.getTime();
????? long l1 = d1.getTime();
????? long l2 = d2.getTime();
????? long difference = l2 - l1;
????? System.out.println("Elapsed milliseconds: " + difference);
?? }
}????
上面的程序打印如下:
Elapsed milliseconds: 1000
這個(gè)程序也帶來(lái)一點(diǎn)混淆。GregorianCalendar類(lèi)的getTime()返回一個(gè)Date對(duì)象,Date類(lèi)的getTime()方法返回從1970年1月1日到這個(gè)時(shí)間的long類(lèi)型的毫秒數(shù)值。雖然他們的方法名字相同,返回值卻不一樣!
下面的程序片斷用簡(jiǎn)單的整數(shù)除法轉(zhuǎn)換毫秒到秒:
long milliseconds = 1999;
long seconds = 1999 / 1000;
這種方法舍去小數(shù)部分轉(zhuǎn)換毫秒到秒,所以1,999毫秒等于1秒,2,000毫秒等于2秒。
計(jì)算更大的單位-例如天數(shù),小時(shí)和分鐘-給定一個(gè)時(shí)間數(shù)值,可以使用下面的過(guò)程:
1. 計(jì)算最大的單位,減去這個(gè)數(shù)值的秒數(shù)
2. 計(jì)算第二大單位,減去這個(gè)數(shù)值的秒數(shù)
3. 重復(fù)操作直到只剩下秒
例如,如果你的時(shí)間的10,000秒,你想知道這個(gè)數(shù)值相應(yīng)的是多少小時(shí),多少分鐘,多少秒,你從最大的單位開(kāi)始:小時(shí)。10,000除以3600(一個(gè)小時(shí)的秒數(shù))得到小時(shí)數(shù)。使用整數(shù)除法,答案是2小時(shí)(整數(shù)除法中小數(shù)舍去)計(jì)算剩下的秒數(shù),10,000-(3,600 x 2) = 2,800秒。所以你有2小時(shí)和2,800秒。
將2,800秒轉(zhuǎn)換成分鐘,2,800除以60。使用整數(shù)除法,答案是46。2,800 - (60 x 46) = 40秒。最后答案是2小時(shí),46分,40秒。
下面的Java程序使用上面的計(jì)算方法:
import java.util.*;
public class Elapsed1 {
?? public void calcHMS(int timeInSeconds) {
????? int hours, minutes, seconds;
????? hours = timeInSeconds / 3600;
????? timeInSeconds = timeInSeconds - (hours * 3600);
????? minutes = timeInSeconds / 60;
????? timeInSeconds = timeInSeconds - (minutes * 60);
????? seconds = timeInSeconds;
????? System.out.println(hours + " hour(s) " + minutes + " minute(s) " + seconds + " second(s)");
?? }
?? public static void main(String[] args) {
????? Elapsed1 elap = new Elapsed1();
????? elap.calcHMS(10000);
?? }
}?
輸出結(jié)果如下:
2 hour(s) 46 minute(s) 40 second(s)
上面的程序甚至在時(shí)間少于一個(gè)小時(shí)也可以正確的計(jì)算小時(shí)數(shù)。例如,你用上面的程序計(jì)算1,000秒,輸出入下:
0 hour(s) 16 minute(s) 40 second(s)
舉一個(gè)現(xiàn)實(shí)世界的例子,下面的程序計(jì)算阿波羅11飛到月球使用得時(shí)間:
import java.util.*;
public class LunarLanding {
?? public long getElapsedSeconds(GregorianCalendar gc1, GregorianCalendar gc2) {
????? Date d1 = gc1.getTime();
????? Date d2 = gc2.getTime();
????? long l1 = d1.getTime();
????? long l2 = d2.getTime();
????? long difference = Math.abs(l2 - l1);
????? return difference / 1000;
?? }
?? public void calcHM(long timeInSeconds) {
????? long hours, minutes, seconds;
????? hours = timeInSeconds / 3600;
????? timeInSeconds = timeInSeconds - (hours * 3600);
????? minutes = timeInSeconds / 60;
????? System.out.println(hours + " hour(s) " + minutes + " minute(s)" );
?? }
?? public static void main(String[] args) {
????? GregorianCalendar lunarLanding = new GregorianCalendar(1969, Calendar.JULY, 20, 16, 17);
????? GregorianCalendar lunarDeparture = new GregorianCalendar(1969, Calendar.JULY, 21, 13, 54);
????? GregorianCalendar startEVA = new GregorianCalendar(1969, Calendar.JULY, 20, 22, 56);
????? GregorianCalendar endEVA = new GregorianCalendar(1969, Calendar.JULY, 21, 1, 9);
????? LunarLanding apollo = new LunarLanding();
????? long eva = apollo.getElapsedSeconds(startEVA, endEVA);
????? System.out.print("EVA duration = ");
????? apollo.calcHM(eva);
????? long lunarStay = apollo.getElapsedSeconds(lunarLanding, lunarDeparture);
????? System.out.print("Lunar stay = ");
????? apollo.calcHM(lunarStay);
?? }
}?????????
上面程序輸出如下:
EVA duration = 2 hour(s) 13 minute(s)
Lunar stay = 21 hour(s) 37 minute(s)
目前為止,我們計(jì)算的基礎(chǔ)公式是這樣的:1分鐘=60秒,1小時(shí)=60分,1天=24小時(shí)。
"1個(gè)月=?天,1年=?天"怎么辦?
月份的天數(shù)有28,29,30,31;一年可以是365或366天。因此,當(dāng)你試圖計(jì)算嚴(yán)格單位的月份和年時(shí),問(wèn)題就產(chǎn)生了。例如,如果你使用月份的平均天數(shù)(近似30.4375),并且計(jì)算下面的時(shí)間間隔:
* July 1, 2:00 a.m. to July 31, 10:00 p.m.
* February 1, 2:00 a.m. to February 29, 10:00 p.m.
第一個(gè)計(jì)算結(jié)果是1個(gè)月;第二個(gè)結(jié)果是0個(gè)月!
所以,在計(jì)算嚴(yán)格單位時(shí)間的月份和年份是要想好。
時(shí)間段,情況 2:時(shí)間單位變化
時(shí)間單位的變化相當(dāng)?shù)暮?jiǎn)單:如果你要統(tǒng)計(jì)天數(shù),你可以簡(jiǎn)單的統(tǒng)計(jì)日期變化次數(shù)。例如,如果某事15日開(kāi)始,17日結(jié)束,經(jīng)過(guò)2天。(日期先是便到16,再到17)同樣的,一個(gè)步驟下午3:25開(kāi)始,4:10 p.m結(jié)束,歷時(shí)1個(gè)小時(shí),因?yàn)樾r(shí)數(shù)值變了一次(從3到4)。
圖書(shū)館經(jīng)常使用這種習(xí)慣計(jì)算時(shí)間。例如,如果你從圖書(shū)館接一本書(shū),我不能占有這本書(shū)最少24小時(shí),會(huì)認(rèn)為圖書(shū)館這樣才給你算一天。而是,我的賬號(hào)上記錄我借書(shū)的日期。日期以變成下一天,我就已經(jīng)結(jié)這本書(shū)一天了,即使總計(jì)不足24小時(shí)。
當(dāng)使用單位的變化來(lái)計(jì)算時(shí)間段,通常感覺(jué)計(jì)算的時(shí)間沒(méi)有多于一個(gè)時(shí)間單位。例如,如果9:00 p.m.我借了一本圖書(shū)館的書(shū),第二天中午還回去,我能算出我借了這本書(shū)一天了。可是,有一種感覺(jué)在問(wèn):"1天和幾個(gè)小時(shí)呢?"這本說(shuō)總計(jì)借出15個(gè)小時(shí),答案是一天還差9個(gè)小時(shí)呢?因此,這篇文章里,我將以一個(gè)時(shí)間單位變化計(jì)算時(shí)間。
單位變化的時(shí)間算法
?這是你怎樣計(jì)算兩個(gè)日期的時(shí)間變化:
1. 制作兩個(gè)日期的拷貝。Close()方法能制作拷貝。
2. 使用日期拷貝,將所有的小于時(shí)間單位變化的部分設(shè)置成它的最小單位。例如,如果計(jì)算天數(shù),那么將小時(shí),分鐘,秒和毫秒設(shè)置成0。這種情況中,使用clear()方法將時(shí)間值設(shè)置稱他們各自的最小值。
3. 取出較早的日期,將你要計(jì)算的單位加1,重復(fù)直到兩個(gè)日期相等。你加1的次數(shù)就是答案。可以使用before()和after()方法,他們返回boolean值,來(lái)判斷是否一個(gè)日期在另一個(gè)日期之前或之后。
下面的類(lèi)的方法用來(lái)計(jì)算天數(shù)和月數(shù)。
import java.util.*;
public class ElapsedTime {
?? public int getDays(GregorianCalendar g1, GregorianCalendar g2) {
????? int elapsed = 0;
????? GregorianCalendar gc1, gc2;
????? if (g2.after(g1)) {
???????? gc2 = (GregorianCalendar) g2.clone();
???????? gc1 = (GregorianCalendar) g1.clone();
????? }
????? else?? {
???????? gc2 = (GregorianCalendar) g1.clone();
???????? gc1 = (GregorianCalendar) g2.clone();
????? }
????? gc1.clear(Calendar.MILLISECOND);
????? gc1.clear(Calendar.SECOND);
????? gc1.clear(Calendar.MINUTE);
????? gc1.clear(Calendar.HOUR_OF_DAY);
????? gc2.clear(Calendar.MILLISECOND);
????? gc2.clear(Calendar.SECOND);
????? gc2.clear(Calendar.MINUTE);
????? gc2.clear(Calendar.HOUR_OF_DAY);
????? while ( gc1.before(gc2) ) {
???????? gc1.add(Calendar.DATE, 1);
???????? elapsed++;
????? }
????? return elapsed;
?? }
?? public int getMonths(GregorianCalendar g1, GregorianCalendar g2) {
????? int elapsed = 0;
????? GregorianCalendar gc1, gc2;
????? if (g2.after(g1)) {
???????? gc2 = (GregorianCalendar) g2.clone();
???????? gc1 = (GregorianCalendar) g1.clone();
????? }
????? else?? {
???????? gc2 = (GregorianCalendar) g1.clone();
???????? gc1 = (GregorianCalendar) g2.clone();
????? }
????? gc1.clear(Calendar.MILLISECOND);
????? gc1.clear(Calendar.SECOND);
????? gc1.clear(Calendar.MINUTE);
????? gc1.clear(Calendar.HOUR_OF_DAY);
????? gc1.clear(Calendar.DATE);
????? gc2.clear(Calendar.MILLISECOND);
????? gc2.clear(Calendar.SECOND);
????? gc2.clear(Calendar.MINUTE);
????? gc2.clear(Calendar.HOUR_OF_DAY);
????? gc2.clear(Calendar.DATE);
????? while ( gc1.before(gc2) ) {
???????? gc1.add(Calendar.MONTH, 1);
???????? elapsed++;
????? }
????? return elapsed;
?? }
}
你可以在上面的類(lèi)中補(bǔ)充另外的方法來(lái)處理小時(shí)和分鐘。同樣,計(jì)算時(shí)間段的算法能更高效一些,尤其是時(shí)間相隔很長(zhǎng)。可是,作為介紹目的,這個(gè)算法有短小和簡(jiǎn)單的優(yōu)勢(shì)。
下面的例子使用ElapsedTime類(lèi)來(lái)計(jì)算兩個(gè)日期之間的天使,而后是月數(shù):
import java.util.*;
public class Example {
?? public static void main(String[] args) {
????? GregorianCalendar gc1 = new GregorianCalendar(2001, Calendar.DECEMBER, 30);
????? GregorianCalendar gc2 = new GregorianCalendar(2002, Calendar.FEBRUARY, 1);
????? ElapsedTime et = new ElapsedTime();
????? int days = et.getDays(gc1, gc2);
????? int months = et.getMonths(gc1, gc2);
????? System.out.println("Days = " + days);
????? System.out.println("Months = " + months);
?? }
}
當(dāng)計(jì)算時(shí),上面的程序可能有用,例如,最近的航班。它顯示下面的輸出:
Days = 33
Months = 2
(OK,關(guān)于航班的計(jì)算有些夸張;這個(gè)天數(shù)算法很適合像圖書(shū)館借書(shū)這樣的應(yīng)用,你看到了她怎樣工作)
告誡
在進(jìn)行時(shí)間工作時(shí)要謹(jǐn)慎:你看到的時(shí)間段的例子,你精確仔細(xì)的考慮非常重要。本文介紹了兩種通常計(jì)算時(shí)間段的想法,但是人們能想到的時(shí)間段的計(jì)算方法僅僅受到人類(lèi)想象力的限制。
所以,當(dāng)寫(xiě)一個(gè)Java程序的時(shí)候,確信你的精確度能讓使用和以來(lái)這些程序的人滿意。同樣,徹底的測(cè)試程序?qū)μ幚頃r(shí)間的程序非重重要。
總結(jié)
本文是在我的前一篇文章 Java時(shí)間計(jì)算介紹怎樣使用GregorianCalendar 和 DateFormat類(lèi)處理時(shí)間問(wèn)題的基礎(chǔ)上的。你已經(jīng)看到了兩種方法來(lái)思考時(shí)間段問(wèn)題和兩種相應(yīng)的途徑使用Java來(lái)處理時(shí)間問(wèn)題。這里提供的信息,很基礎(chǔ),提供給你一個(gè)在Java中處理時(shí)間問(wèn)題的有力工具。
關(guān)于作者
?Robert Nielsen是SCJP。他擁有碩士學(xué)位,專(zhuān)攻計(jì)算機(jī)教育,并且在計(jì)算機(jī)領(lǐng)域執(zhí)教多年。他也在各樣的雜志上發(fā)表過(guò)很多計(jì)算機(jī)相關(guān)的文章。
關(guān)于譯者
Cocia Lin(cocia@163.com)是程序員。它擁有學(xué)士學(xué)位,現(xiàn)在專(zhuān)攻Java相關(guān)技術(shù),剛剛開(kāi)始在計(jì)算機(jī)領(lǐng)域折騰。
java 中的時(shí)間操作不外乎這四種情況:
1 、獲取當(dāng)前時(shí)間
2 、獲取某個(gè)時(shí)間的某種格式
3 、設(shè)置時(shí)間
4 、時(shí)間的運(yùn)算
好,下面就針對(duì)這三種情況,一個(gè)一個(gè)搞定。
?
一、獲取當(dāng)前時(shí)間
?
有兩種方式可以獲得,第一種,使用 Date 類(lèi)。
j2SE 的包里有兩個(gè) Date 類(lèi),一個(gè)是 java.sql.Date, 一個(gè)是 java.util.Date
這里,要使用 java.util.Date 。獲取當(dāng)前時(shí)間的代碼如下
Date date = new Date();
date.getTime() ;
還有一種方式,使用 System.currentTimeMillis() ;
?
這兩種方式獲得的結(jié)果是一樣的,都是得到一個(gè)當(dāng)前的時(shí)間的 long 型的時(shí)間的毫秒值,這個(gè)值實(shí)際上是當(dāng)前時(shí)間值與 1970 年一月一號(hào)零時(shí)零分零秒相差的毫秒數(shù)。
當(dāng)前的時(shí)間得到了,但實(shí)際的應(yīng)用中最后往往不是要用這個(gè) long 型的東西,用戶希望得到的往往是一個(gè)時(shí)間的字符串,比如“ 2006 年 6 月 18 號(hào)”,或“ 2006-06-18 ”,老外可能希望得到的是“ 06-18-2006 ”,諸如此類(lèi)等等。這就是下一個(gè)要解決的問(wèn)題
?
二、獲取某個(gè)時(shí)間的某種格式
?
獲取時(shí)間的格式,需要用到一個(gè)專(zhuān)門(mén)用于時(shí)間格式的類(lèi) java.text.SimpleDateFormat 。
首先,定義一個(gè) SimpleDateFormat 變量
SimpleDateFormat sdf = new SimpleDateFormat("",Locale.SIMPLIFIED_CHINESE);
這個(gè)構(gòu)造函數(shù)的定義如下:
SimpleDateFormat(String ? pattern, Locale ? locale)
第一個(gè)參數(shù) pattern ,我們后面再解釋?zhuān)@里我們使用一個(gè) "", 第二個(gè)參數(shù),是用來(lái)設(shè)置時(shí)區(qū)的,這里用到了 java.util.Locale 這個(gè)類(lèi),這個(gè)類(lèi)了面定義了很多靜態(tài)變量,直接拿過(guò)來(lái)用就 OK ,我們把時(shí)區(qū)設(shè)置為 Locale.SIMPLIFIED_CHINESE ,只看名字,這個(gè)靜態(tài)變量的意義已經(jīng)很清楚了。
?
接下來(lái)我們使用這個(gè) SimpleDateFormat 把當(dāng)前時(shí)間格式化為一個(gè)如下格式的時(shí)間字符串“ XXXX 年 XX 月 XX 日 _XX 時(shí) XX 分 XX 秒”,代碼:
sdf.applyPattern("yyyy年MM月dd日_HH時(shí)mm分ss秒");
String timeStr = sdf.format(new Date());
?
獲取時(shí)間格式的函數(shù)是 format ,這個(gè)函數(shù)的參數(shù)是 java.util.Date 對(duì)象,這個(gè)沒(méi)有什么花頭。
要說(shuō)明一下的是這個(gè) pattern ,所謂的模式。這里, yyyy,MM,dd 等,這就是模式。
我們可以在 SimpleDateFormat 的構(gòu)造函數(shù)中指定模式,比如
SimpleDateFormat sdf = new SimpleDateFormat(" yyyy-MM-dd ",Locale.SIMPLIFIED_CHINESE);
也可以獲取時(shí)間格式的時(shí)候使用applyPattern函數(shù)臨時(shí)指定,上面的例子就是這樣。
什么字符代表什么,這是 j2se 約定好的,設(shè)置模式的時(shí)候,我們可以使用約定好的字符加上任何我們想要的字符串。
j2se 對(duì)字符所代表的模式的約定列表如下:
?
?
Letter
Date or Time Component
Presentation
G
Era designator
Text
y
Year
Year
M
Month in year
Month
w
Week in year
Number
W
Week in month
Number
D
Day in year
Number
d
Day in month
Number
F
Day of week in month
Number
E
Day in week
Text
a
Am/pm marker
Text?
H
Hour in day (0-23)
Number
k
Hour in day (1-24)
Number
K
Hour in am/pm (0-11)?
Number
h??
Hour in am/pm (1-12)?
Number
m??
Minute in hour?
Number
s??
Second in minute?
Number
S??
Millisecond?
Number??
z??
Time zone??
General time zone?
Z??
Time zone?
RFC 822 time zone?
?
posted on 2006-07-04 09:13 liaojiyong 閱讀(3347) 評(píng)論(0) 編輯 收藏 所屬分類(lèi): Java