計算Java日期
學(xué)習(xí)怎樣創(chuàng)建和使用日期
概要
??? 不管你是處理財務(wù)交易還是計劃著下一步的行動,你都要知道怎樣在Java中建立,使用和顯示日期。這需要你簡單的查閱一下相應(yīng)類的API參考:一個日期可以創(chuàng)建3個相關(guān)類的對象。這篇文章告訴你你想要知道的內(nèi)容。(3,000字)
作者:Robert Nielsen
翻譯:Cocia Lin
??? Java統(tǒng)計從1970年1月1日起的毫秒的數(shù)量表示日期。也就是說,例如,1970年1月2日,是在1月1日后的86,400,000毫秒。同樣的,1969年12月31日是在1970年1月1日前86,400,000毫秒。Java的Date類使用long類型紀(jì)錄這些毫秒值.因?yàn)閘ong是有符號整數(shù),所以日期可以在1970年1月1日之前,也可以在這之后。Long類型表示的最大正值和最大負(fù)值可以輕松的表示290,000,000年的時間,這適合大多數(shù)人的時間要求。
Date 類
?? Date類可以在java.util包中找到,用一個long類型的值表示一個指定的時刻。它的一個有用的構(gòu)造函數(shù)是Date(),它創(chuàng)建一個表示創(chuàng)建時刻的對象。getTime()方法返回Date對象的long值。在下面的程序中,我使用Date()構(gòu)造函數(shù)創(chuàng)建一個表示程序運(yùn)行時刻的對象,并且利用getTime()方法找到這個日期代表的毫秒數(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)行這個程序后,我得到972,568,255,150.快速確認(rèn)一下這個數(shù)字,起碼在一個合理的范圍:它不到31年,這個數(shù)值相對1970年1月1日到我寫這篇文章的時間來說,是合理的。計算機(jī)是這個毫秒值表示時間,人們可不愿意說" 我將在996,321,998,34見到你。"幸運(yùn)的是,Java提供了一個轉(zhuǎn)換Date對象到字符串的途徑,表示成傳統(tǒng)的形式。我們在下一節(jié)討論DateFormat類,它直觀的建立日期字符串。
DateFormat類
?DateFormat類的一個目標(biāo)是建立一個人們能夠識別的字符串。然而,因?yàn)檎Z言的差別,不是所有的人希望看到嚴(yán)格的相同格式的日期。法國人更喜歡看到"25 decembre 2000,",但是美國人習(xí)慣看到"December 25,2000."所以一個DateFormat的實(shí)例創(chuàng)建以后,這個對象包含了日期的顯示格式的信息。如果使用用戶電腦區(qū)域設(shè)置缺省的格式,你可以象下面那樣,創(chuàng)建DateFormat對象,使用getDateInstance()方法:
DateFormat df = DateFormat.getDateInstance();??
DateFormat類在java.text包中可以找到。
轉(zhuǎn)換成字符串
你可以使用format()方法轉(zhuǎn)換Date對象為一個字符串。下面的示例程序說明了這個問題:
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);
?? }
}?
在上面的代碼中,展示了沒有參數(shù),使用缺省格式的getDateInstance()方法。Java還提供了幾個選擇日期格式,你可以通過使用重載的getDateInstance(int style)獲得。出于方便的原因,DateFormat提供了幾種預(yù)置的常量,你可以使用這些常量參數(shù)。下面是幾個SHORT, MEDIUM, LONG, 和FULL類型的示例:
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????
?
從這里,你能看到,瑞典的月份不是大寫的(雖然November還是november).還有,LONG和FULL版本在瑞典語中是一樣的,但是美國英語卻不同。另外,有趣的是,瑞典語單詞的星期三,onsdag,沒有包含在FULL日期里,英語卻包括。
注意你能夠使用getDateInstance()方法改變DateFormat實(shí)例的語種;但是,在上面的例子中,是通過改變Windows98的控制面板的區(qū)域設(shè)置做到的。不同的地方的區(qū)域設(shè)置不同,結(jié)果就不同,這樣有好處,也有不足,Java程序員應(yīng)該了解這些。一個好處是Java程序員可以只寫一行代碼就可以顯示日期,而且世界不同地區(qū)的電腦運(yùn)行同樣的程序會有不用的日期格式。 但是這也是一個缺點(diǎn),當(dāng)程序員希望顯示同一種格式的時--這也有可取之處,舉例來說,在程序中混合輸出文本和日期,如果文本是英文,我們就不希望日期格式是其他的格式,象德文或是西班牙文。如果程序員依靠日期格式編程,日期格式將根據(jù)運(yùn)行程序所在電腦的區(qū)域設(shè)置不用而不同。
解析字符串
?通過parse()方法,DateFormat能夠以一個字符串創(chuàng)立一個Date對象。這個方法能拋出ParseException異常,所以你必須使用適當(dāng)?shù)漠惓L幚砑夹g(shù)。下面的例子程序通過字符串創(chuàng)建Date對象:
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)建一個任意的日期時parse()方法很有用。我將通過另一種方法創(chuàng)建一個任意得日期。同時,你將看到怎樣進(jìn)行基本日期計算,例如計算90天后的另一天。你可以使用GregorianCalendar類來完成這個任務(wù)。
GregorianCalendar類
?創(chuàng)建一個代表任意日期的一個途徑使用GregorianCalendar類的構(gòu)造函數(shù),它包含在java.util包中:
GregorianCalendar(int year, int month, int date)
注意月份的表示,一月是0,二月是1,以此類推,是12月是11。因?yàn)榇蠖鄶?shù)人習(xí)慣于使用單詞而不是使用數(shù)字來表示月份,這樣程序也許更易讀,父類Calendar使用常量來表示月份:JANUARY, FEBRUARY,等等。所以,創(chuàng)建Wilbur 和 Orville制造第一架動力飛機(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對象到字符串。這里,你可以做同樣的事情;但是首先,你需要將GregorianCalendar對象轉(zhuǎn)換到Date。要做到這一點(diǎn),你可以使用getTime()方法,從它得父類Calendar繼承而來。GetTime()方法返回GregorianCalendar相應(yīng)的Date對象。你能夠創(chuàng)建GregorianCalendar對象,轉(zhuǎn)換到Date對象,得到和輸出相應(yīng)的字符串這樣一個過程。下面是例子:
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);
?? }
}
有時候創(chuàng)建一個代表當(dāng)前時刻的GregorianCalendar類的實(shí)例是很有用的。你可以簡單的使用沒有參數(shù)的GregorianCalendar構(gòu)造函數(shù),象這樣:
GregorianCalendar thisday = new GregorianCalendar();
一個輸出今天日期的例子程序,使用GregorianCalendar對象:
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ù)很類似:都創(chuàng)建一個對象,條件簡單,代表今天。
日期處理
GregorianCalendar類提供處理日期的方法。一個有用的方法是add().使用add()方法,你能夠增加象年,月數(shù),天數(shù)到日期對象中。要使用add()方法,你必須提供要增加的字段,要增加的數(shù)量。一些有用的字段是DATE, MONTH, YEAR, 和 WEEK_OF_YEAR。下面的程序使用add()方法計算未來80天的一個日期。在Jules的<環(huán)球80天>是一個重要的數(shù)字,使用這個程序可以計算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);
?? }
}
這個例子是想象的,但在一個日期上增加天數(shù)是一個普遍的操作:影碟可以租3天,圖書館可以借書21天,商店經(jīng)常需要將購買的物品在30天內(nèi)賣出。下面的程序演示了使用年計算:
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()一個重要的副作用是它改變的原來的日期。有時候,擁有原始日期和修改后的日期很重要。不幸的是,你不能簡單的創(chuàng)建一個GregorianCalendar對象,設(shè)置它和原來的相等(equal)。原因是兩個變量指向同一個Date()對象地址。如果Date對象改變,兩個變量就指向改變后的日期對象。代替這種做法,應(yīng)該創(chuàng)建一個新對象。下面的程序示范了這種做法:
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)閮蓚€對象指向同一個Date,而Date已經(jīng)被改變了)。對象gc3指向一個單獨(dú)的Date,它沒有被改變。
計算復(fù)習(xí)日期
在這節(jié),你將看到一個依據(jù)現(xiàn)實(shí)世界的例子。這個詳細(xì)的程序計算過去一個具體的日期。例如,你閱讀這篇文章,你想要記住一個印象深刻的知識點(diǎn)。如果你沒有照片一樣的記憶力,你就要定期的復(fù)習(xí)這些新資料,這將幫助你記住它。關(guān)于復(fù)習(xí)系統(tǒng),Kurt Hanks 和 Gerreld L. Pulsipher在他們的< Five Secrets to Personal Productivity個人能力的5個秘密>中有討論,建議看過第一眼后馬上回顧一下,然后是1天后,1個星期后,1個月后,3個月后,1年后。我的這篇文章,你要馬上回顧一下,從現(xiàn)在算起,再就是明天,然后是1個星期,1個月,3個月,1年后。我們的程序?qū)⒂嬎氵@些日期。
這個程序非常有用的,它將是PIM(Personal Information Manager個人信息管理器)的一個組成部分,并將確定復(fù)習(xí)時間。在下面的程序中,getDates()方法對一個返回日期數(shù)組(復(fù)習(xí)日期)的電子軟件很有用。另外,你可以返回單獨(dú)的一個日期,使用getFirstDay(),getOneDay(),getOneWeek(),getOnMonth()和getOneYear().當(dāng)時間范圍超出這個PIM的ReviewDates的計算范圍時ReviewDates類演示了怎樣計算時間段。現(xiàn)在,你可以容易的修改它用來處理你需要的時間段,象圖書館借書,錄影帶租賃和抵押計算。首先,ReviewDates類顯示在下面:
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類列出復(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個重要的類:Date,DateFormat,GregorianCalendar.這些類讓你創(chuàng)建日期,轉(zhuǎn)換成字符串,和計算日期基本元素。處理Java中的日期問題,這篇文章只是冰山一角。可是,我在這里介紹的類和方法不僅僅是你學(xué)習(xí)高級技術(shù)的跳板,這些類和方法本身就可以處理很多通常的日期相關(guān)的任務(wù)