<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    love fish大鵬一曰同風起,扶搖直上九萬里

    常用鏈接

    統計

    積分與排名

    friends

    link

    最新評論

    Java的時間處理

    計算Java日期
    學習怎樣創建和使用日期
    概要
    ? ? 不管你是處理財務交易還是計劃著下一步的行動,你都要知道怎樣在Java中建立,使用和顯示日期。這需要你簡單的查閱一下相應類的API參考:一個日期可以創建3個相關類的對象。這篇文章告訴你你想要知道的內容。(3,000字)
    作者:Robert Nielsen
    翻譯:Cocia Lin
    轉帖:蕭十一郎

    ? ? Java統計從1970年1月1日起的毫秒的數量表示日期。也就是說,例如,1970年1月2日,是在1月1日后的86,400,000毫秒。同樣的,1969年12月31日是在1970年1月1日前86,400,000毫秒。Java的Date類使用long類型紀錄這些毫秒值.因為long是有符號整數,所以日期可以在1970年1月1日之前,也可以在這之后。Long類型表示的最大正值和最大負值可以輕松的表示290,000,000年的時間,這適合大多數人的時間要求。
    Date 類
    ? ?Date類可以在java.util包中找到,用一個long類型的值表示一個指定的時刻。它的一個有用的構造函數是Date(),它創建一個表示創建時刻的對象。getTime()方法返回Date對象的long值。在下面的程序中,我使用Date()構造函數創建一個表示程序運行時刻的對象,并且利用getTime()方法找到這個日期代表的毫秒數量:

    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);
    ? ?}
    }

    當我運行這個程序后,我得到972,568,255,150.快速確認一下這個數字,起碼在一個合理的范圍:它不到31年,這個數值相對1970年1月1日到我寫這篇文章的時間來說,是合理的。計算機是這個毫秒值表示時間,人們可不愿意說" 我將在996,321,998,34見到你。"幸運的是,Java提供了一個轉換Date對象到字符串的途徑,表示成傳統的形式。我們在下一節討論DateFormat類,它直觀的建立日期字符串。
    DateFormat類
    DateFormat類的一個目標是建立一個人們能夠識別的字符串。然而,因為語言的差別,不是所有的人希望看到嚴格的相同格式的日期。法國人更喜歡看到"25 decembre 2000,",但是美國人習慣看到"December 25,2000."所以一個DateFormat的實例創建以后,這個對象包含了日期的顯示格式的信息。如果使用用戶電腦區域設置缺省的格式,你可以象下面那樣,創建DateFormat對象,使用getDateInstance()方法:

    DateFormat df = DateFormat.getDateInstance();? ?

    DateFormat類在java.text包中可以找到。
    轉換成字符串
    你可以使用format()方法轉換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);
    ? ?}
    }??

    在上面的代碼中,展示了沒有參數,使用缺省格式的getDateInstance()方法。Java還提供了幾個選擇日期格式,你可以通過使用重載的getDateInstance(int style)獲得。出于方便的原因,DateFormat提供了幾種預置的常量,你可以使用這些常量參數。下面是幾個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

    同樣的程序,在我的電腦上使用缺省設置運行后,改變區域設置為瑞典,輸出如下:

    (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實例的語種;但是,在上面的例子中,是通過改變Windows98的控制面板的區域設置做到的。不同的地方的區域設置不同,結果就不同,這樣有好處,也有不足,Java程序員應該了解這些。一個好處是Java程序員可以只寫一行代碼就可以顯示日期,而且世界不同地區的電腦運行同樣的程序會有不用的日期格式。 但是這也是一個缺點,當程序員希望顯示同一種格式的時--這也有可取之處,舉例來說,在程序中混合輸出文本和日期,如果文本是英文,我們就不希望日期格式是其他的格式,象德文或是西班牙文。如果程序員依靠日期格式編程,日期格式將根據運行程序所在電腦的區域設置不用而不同。
    解析字符串
    通過parse()方法,DateFormat能夠以一個字符串創立一個Date對象。這個方法能拋出ParseException異常,所以你必須使用適當的異常處理技術。下面的例子程序通過字符串創建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);
    ? ?? ?}
    ? ?}
    }

    在創建一個任意的日期時parse()方法很有用。我將通過另一種方法創建一個任意得日期。同時,你將看到怎樣進行基本日期計算,例如計算90天后的另一天。你可以使用GregorianCalendar類來完成這個任務。
    GregorianCalendar類
    創建一個代表任意日期的一個途徑使用GregorianCalendar類的構造函數,它包含在java.util包中:

    GregorianCalendar(int year, int month, int date)

    注意月份的表示,一月是0,二月是1,以此類推,是12月是11。因為大多數人習慣于使用單詞而不是使用數字來表示月份,這樣程序也許更易讀,父類Calendar使用常量來表示月份:JANUARY, FEBRUARY,等等。所以,創建Wilbur 和 Orville制造第一架動力飛機的日期(December 17, 1903),你可以使用:

    GregorianCalendar firstFlight = new GregorianCalendar(1903, Calendar.DECEMBER, 17);??
    出于清楚的考慮,你應該使用前面的形式。但是,你也應該學習怎樣閱讀下面的短格式。下面的例子同樣表示December 17,1903(記住,在短格式中,11表示December)

    GregorianCalendar firstFlight = new GregorianCalendar(1903, 11, 17);? ?

    在上一節中,你學習了轉換Date對象到字符串。這里,你可以做同樣的事情;但是首先,你需要將GregorianCalendar對象轉換到Date。要做到這一點,你可以使用getTime()方法,從它得父類Calendar繼承而來。GetTime()方法返回GregorianCalendar相應的Date對象。你能夠創建GregorianCalendar對象,轉換到Date對象,得到和輸出相應的字符串這樣一個過程。下面是例子:

    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);
    ? ?}
    }

    有時候創建一個代表當前時刻的GregorianCalendar類的實例是很有用的。你可以簡單的使用沒有參數的GregorianCalendar構造函數,象這樣:

    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()構造函數和GregorianCalendar()構造函數很類似:都創建一個對象,條件簡單,代表今天。
    日期處理
    GregorianCalendar類提供處理日期的方法。一個有用的方法是add().使用add()方法,你能夠增加象年,月數,天數到日期對象中

    R褂胊dd()方法,你必須提供要增加的字段,要增加的數量。一些有用的字段是DATE, MONTH, YEAR, 和 WEEK_OF_YEAR。下面的程序使用add()方法計算未來80天的一個日期。在Jules的<環球80天>是一個重要的數字,使用這個程序可以計算Phileas Fogg從出發的那一天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);
    ? ?}
    }

    這個例子是想象的,但在一個日期上增加天數是一個普遍的操作:影碟可以租3天,圖書館可以借書21天,商店經常需要將購買的物品在30天內賣出。下面的程序演示了使用年計算:

    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()一個重要的副作用是它改變的原來的日期。有時候,擁有原始日期和修改后的日期很重要。不幸的是,你不能簡單的創建一個GregorianCalendar對象,設置它和原來的相等(equal)。原因是兩個變量指向同一個Date()對象地址。如果Date對象改變,兩個變量就指向改變后的日期對象。代替這種做法,應該創建一個新對象。下面的程序示范了這種做法:

    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);
    ? ?}
    }

    ? ? 程序運行后,gc1和gc2被變成2001年(因為兩個對象指向同一個Date,而Date已經被改變了)。對象gc3指向一個單獨的Date,它沒有被改變。
    計算復習日期
    在這節,你將看到一個依據現實世界的例子。這個詳細的程序計算過去一個具體的日期。例如,你閱讀這篇文章,你想要記住一個印象深刻的知識點。如果你沒有照片一樣的記憶力,你就要定期的復習這些新資料,這將幫助你記住它。關于復習系統,Kurt Hanks 和 Gerreld L. Pulsipher在他們的< Five Secrets to Personal Productivity個人能力的5個秘密>中有討論,建議看過第一眼后馬上回顧一下,然后是1天后,1個星期后,1個月后,3個月后,1年后。我的這篇文章,你要馬上回顧一下,從現在算起,再就是明天,然后是1個星期,1個月,3個月,1年后。我們的程序將計算這些日期。
    這個程序非常有用的,它將是PIM(Personal Information Manager個人信息管理器)的一個組成部分,并將確定復習時間。在下面的程序中,getDates()方法對一個返回日期數組(復習日期)的電子軟件很有用。另外,你可以返回單獨的一個日期,使用getFirstDay(),getOneDay(),getOneWeek(),getOnMonth()和getOneYear().當時間范圍超出這個PIM的ReviewDates的計算范圍時ReviewDates類演示了怎樣計算時間段。現在,你可以容易的修改它用來處理你需要的時間段,象圖書館借書,錄影帶租賃和抵押計算。首先,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類列出復習日期的例子程序:

    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();
    ? ?}
    }

    總結
    這篇文章介紹了關于日期處理的3個重要的類:Date,DateFormat,GregorianCalendar.這些類讓你創建日期,轉換成字符串,和計算日期基本元素。處理Java中的日期問題,這篇文章只是冰山一角。可是,我在這里介紹的類和方法不僅僅是你學習高級技術的跳板,這些類和方法本身就可以處理很多通常的日期相關的任務
    關于作者
    Robert Nielsen是SCJP。他擁有碩士學位,專攻計算機教育,并且在計算機領域執教多年


    1. Java計算時間依靠1970年1月1日開始的毫秒數.??????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????
    2. Date類的構造函數Date()返回代表當前創建的時刻的對象。Date的方法getTime()返回一個long值在數值上等于1970年1月1日之前或之后的時刻。
    3. DateFormat類用來轉換Date到String,反之亦然。靜態方法getDateInstance()返回DateFormat的缺省格式;getDateInstance(DateFormat.FIELD)返回指定的DateFormat對象格式。Format(Date d)方法返回String表示日期,例如"January 1,2002."反過來,parse(String s)方法返回以參數字符串表示的Date對象。
    4. format()方法返回的字符串格式根據不同地區的時間設置而有所不同。
    5. GregorianCalendear類有兩個重要的構造函數:GregorianCalerdar(),返回代表當前創建時間的對象;GregorianCalendar(int year,int month,int date)返回代表任意日期的對象。GregorianCalendar類的getTime()方法返回日期對象。Add(int field,int amount)方法通過加或減時間單位,象天數,月數或年數來計算日期。
    GregorianCalendar和 時間
    ?兩個GregorianCalendar的構造函數可以用來處理時間。前者創建一個表示日期,小時和分鐘的對象:

    GregorianCalendar(int year, int month, int date, int hour, int minute)

    第二個創建一個表示一個日期,小時,分鐘和秒:

    GregorianCalendar(int year, int month, int date, int hour, int minute, int second)

    首先,我應該提醒一下,每一個構造函數需要時間信息中的日期信息(年,月,日)。如果你想說2:30 p.m.,你必須指出日期。
    同樣,每一個GregorianCalendar構造函數創建一個在時間上使用毫秒計算的對象。所以,如果你的構造函數只提供年,月,日參數,那小時,分鐘,秒和毫秒的值將被置0.
    DateFormat和時間
    你可以使用靜態方法getDateTimeInstance(int dateStyle,int timeStyle)來建立DateFormat對象來顯示時間和日期。這個方法表明你想要的日期和時間格式。如果你喜歡使用缺省格式,可以使用getDateTimeInstance()來代替它。
    你可以使用靜態方法getTimeInstance(int timeStyle)創建DateFormat對象來顯示正確的時間。
    下面的程序示范了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
    (輸出根據你所在得地區有所不同)

    計算時間間隔
    ???? 你可能有時需要計算過去的時間;例如,給你開始和結束時間,你想知道制造流程的持續時間。一個出租公司按小時或天數出租東西,計算時間對他們也很有用。同樣的,在金融界,經常需要計算重要的支付時間。
    將問題復雜化,人類至少是用兩種方法計算時間。你可以說一天已經結束當24小時過去了,或者日歷從今天翻到明天。我們將討論我們想到的這兩種情況。
    時間段,情況 1:嚴格時間單位
    在這種情況中,只有24小時過去,這天才過去,60分鐘過去,這個小時才過去,60秒過去,這個分鐘才過去,以此類推。在這個方法中,23小時的時間將被認為是0天。
    使用這種方法計算時間段,你從計算過去的毫秒開始。為了做到這一點,首先轉換每個日期為從1970年1月1日起得毫秒數。你可以從第二個毫秒值中減去第一個毫秒值。這里有一個簡單的計算:

    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

    這個程序也帶來一點混淆。GregorianCalendar類的getTime()返回一個Date對象,Date類的getTime()方法返回從1970年1月1日到這個時間的long類型的毫秒數值。雖然他們的方法名字相同,返回值卻不一樣!
    下面的程序片斷用簡單的整數除法轉換毫秒到秒:

    long milliseconds = 1999;
    long seconds = 1999 / 1000;

    這種方法舍去小數部分轉換毫秒到秒,所以1,999毫秒等于1秒,2,000毫秒等于2秒。
    計算更大的單位-例如天數,小時和分鐘-給定一個時間數值,可以使用下面的過程:
    1. 計算最大的單位,減去這個數值的秒數
    2. 計算第二大單位,減去這個數值的秒數
    3. 重復操作直到只剩下秒
    例如,如果你的時間的10,000秒,你想知道這個數值相應的是多少小時,多少分鐘,多少秒,你從最大的單位開始:小時。10,000除以3600(一個小時的秒數)得到小時數。使用整數除法,答案是2小時(整數除法中小數舍去)計算剩下的秒數,10,000-(3,600 x 2) = 2,800秒。所以你有2小時和2,800秒。
    將2,800秒轉換成分鐘,2,800除以60。使用整數除法,答案是46。2,800 - (60 x 46) = 40秒。最后答案是2小時,46分,40秒。
    下面的Java程序使用上面的計算方法:

    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);
    ?? }
    }?

    輸出結果如下:

    2 hour(s) 46 minute(s) 40 second(s)

    上面的程序甚至在時間少于一個小時也可以正確的計算小時數。例如,你用上面的程序計算1,000秒,輸出入下:
    0 hour(s) 16 minute(s) 40 second(s)
    舉一個現實世界的例子,下面的程序計算阿波羅11飛到月球使用得時間:

    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)

    目前為止,我們計算的基礎公式是這樣的:1分鐘=60秒,1小時=60分,1天=24小時。
    "1個月=?天,1年=?天"怎么辦?
    月份的天數有28,29,30,31;一年可以是365或366天。因此,當你試圖計算嚴格單位的月份和年時,問題就產生了。例如,如果你使用月份的平均天數(近似30.4375),并且計算下面的時間間隔:

    * 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.

    第一個計算結果是1個月;第二個結果是0個月!
    所以,在計算嚴格單位時間的月份和年份是要想好。
    時間段,情況 2:時間單位變化
    時間單位的變化相當的簡單:如果你要統計天數,你可以簡單的統計日期變化次數。例如,如果某事15日開始,17日結束,經過2天。(日期先是便到16,再到17)同樣的,一個步驟下午3:25開始,4:10 p.m結束,歷時1個小時,因為小時數值變了一次(從3到4)。
    圖書館經常使用這種習慣計算時間。例如,如果你從圖書館接一本書,我不能占有這本書最少24小時,會認為圖書館這樣才給你算一天。而是,我的賬號上記錄我借書的日期。日期以變成下一天,我就已經結這本書一天了,即使總計不足24小時。
    當使用單位的變化來計算時間段,通常感覺計算的時間沒有多于一個時間單位。例如,如果9:00 p.m.我借了一本圖書館的書,第二天中午還回去,我能算出我借了這本書一天了。可是,有一種感覺在問:"1天和幾個小時呢?"這本說總計借出15個小時,答案是一天還差9個小時呢?因此,這篇文章里,我將以一個時間單位變化計算時間。
    單位變化的時間算法
    ?這是你怎樣計算兩個日期的時間變化:
    1. 制作兩個日期的拷貝。Close()方法能制作拷貝。
    2. 使用日期拷貝,將所有的小于時間單位變化的部分設置成它的最小單位。例如,如果計算天數,那么將小時,分鐘,秒和毫秒設置成0。這種情況中,使用clear()方法將時間值設置稱他們各自的最小值。
    3. 取出較早的日期,將你要計算的單位加1,重復直到兩個日期相等。你加1的次數就是答案。可以使用before()和after()方法,他們返回boolean值,來判斷是否一個日期在另一個日期之前或之后。
    下面的類的方法用來計算天數和月數。

    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;
    ?? }
    }

    你可以在上面的類中補充另外的方法來處理小時和分鐘。同樣,計算時間段的算法能更高效一些,尤其是時間相隔很長。可是,作為介紹目的,這個算法有短小和簡單的優勢。
    下面的例子使用ElapsedTime類來計算兩個日期之間的天使,而后是月數:

    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);
    ?? }
    }

    當計算時,上面的程序可能有用,例如,最近的航班。它顯示下面的輸出:

    Days = 33
    Months = 2

    (OK,關于航班的計算有些夸張;這個天數算法很適合像圖書館借書這樣的應用,你看到了她怎樣工作)
    告誡
    在進行時間工作時要謹慎:你看到的時間段的例子,你精確仔細的考慮非常重要。本文介紹了兩種通常計算時間段的想法,但是人們能想到的時間段的計算方法僅僅受到人類想象力的限制。
    所以,當寫一個Java程序的時候,確信你的精確度能讓使用和以來這些程序的人滿意。同樣,徹底的測試程序對處理時間的程序非重重要。
    總結
    本文是在我的前一篇文章 Java時間計算介紹怎樣使用GregorianCalendar 和 DateFormat類處理時間問題的基礎上的。你已經看到了兩種方法來思考時間段問題和兩種相應的途徑使用Java來處理時間問題。這里提供的信息,很基礎,提供給你一個在Java中處理時間問題的有力工具。

    關于作者
    ?Robert Nielsen是SCJP。他擁有碩士學位,專攻計算機教育,并且在計算機領域執教多年。他也在各樣的雜志上發表過很多計算機相關的文章。
    關于譯者
    Cocia Lin(cocia@163.com)是程序員。它擁有學士學位,現在專攻Java相關技術,剛剛開始在計算機領域折騰。

    java 中的時間操作不外乎這四種情況:

    1 、獲取當前時間

    2 、獲取某個時間的某種格式

    3 、設置時間

    4 、時間的運算

    好,下面就針對這三種情況,一個一個搞定。

    ?

    一、獲取當前時間

    ?

    有兩種方式可以獲得,第一種,使用 Date 類。

    j2SE 的包里有兩個 Date 類,一個是 java.sql.Date, 一個是 java.util.Date

    這里,要使用 java.util.Date 。獲取當前時間的代碼如下

    Date date = new Date();

    date.getTime() ;

    還有一種方式,使用 System.currentTimeMillis() ;

    ?

    這兩種方式獲得的結果是一樣的,都是得到一個當前的時間的 long 型的時間的毫秒值,這個值實際上是當前時間值與 1970 年一月一號零時零分零秒相差的毫秒數。

    當前的時間得到了,但實際的應用中最后往往不是要用這個 long 型的東西,用戶希望得到的往往是一個時間的字符串,比如“ 2006 6 18 號”,或“ 2006-06-18 ”,老外可能希望得到的是“ 06-18-2006 ”,諸如此類等等。這就是下一個要解決的問題

    ?

    二、獲取某個時間的某種格式

    ?

    獲取時間的格式,需要用到一個專門用于時間格式的類 java.text.SimpleDateFormat

    首先,定義一個 SimpleDateFormat 變量

    SimpleDateFormat sdf = new SimpleDateFormat("",Locale.SIMPLIFIED_CHINESE);

    這個構造函數的定義如下:

    SimpleDateFormat(String ? pattern, Locale ? locale)

    第一個參數 pattern ,我們后面再解釋,這里我們使用一個 "", 第二個參數,是用來設置時區的,這里用到了 java.util.Locale 這個類,這個類了面定義了很多靜態變量,直接拿過來用就 OK ,我們把時區設置為 Locale.SIMPLIFIED_CHINESE ,只看名字,這個靜態變量的意義已經很清楚了。

    ?

    接下來我們使用這個 SimpleDateFormat 把當前時間格式化為一個如下格式的時間字符串“ XXXX XX XX _XX XX XX 秒”,代碼:

    sdf.applyPattern("yyyy年MM月dd日_HH時mm分ss秒");

    String timeStr = sdf.format(new Date());

    ?

    獲取時間格式的函數是 format ,這個函數的參數是 java.util.Date 對象,這個沒有什么花頭。

    要說明一下的是這個 pattern ,所謂的模式。這里, yyyy,MM,dd 等,這就是模式。

    我們可以在 SimpleDateFormat 的構造函數中指定模式,比如

    SimpleDateFormat sdf = new SimpleDateFormat(" yyyy-MM-dd ",Locale.SIMPLIFIED_CHINESE);

    也可以獲取時間格式的時候使用applyPattern函數臨時指定,上面的例子就是這樣。

    什么字符代表什么,這是 j2se 約定好的,設置模式的時候,我們可以使用約定好的字符加上任何我們想要的字符串。

    j2se 對字符所代表的模式的約定列表如下:

    ?

    ?

    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 閱讀(3348) 評論(0)  編輯  收藏 所屬分類: Java

    主站蜘蛛池模板: 亚洲中文字幕无码mv| 久久免费精品一区二区| 曰韩亚洲av人人夜夜澡人人爽 | 青青草国产免费国产是公开| 337p日本欧洲亚洲大胆裸体艺术| 免费不卡在线观看AV| 亚洲jizzjizz少妇| 亚洲国产AV无码专区亚洲AV| 一个人在线观看视频免费| 一级特黄色毛片免费看| 亚洲成a人片77777群色| 亚洲成av人片不卡无码久久| 7x7x7x免费在线观看| 色屁屁www影院免费观看视频| 亚洲av之男人的天堂网站| 免费涩涩在线视频网| 欧洲人成在线免费| 小说区亚洲自拍另类| 自怕偷自怕亚洲精品| 免费在线观看黄色毛片| 日本免费一区二区在线观看| www在线观看免费视频| 亚洲国产乱码最新视频| 亚洲日本va中文字幕久久| 日本视频免费在线| 4399影视免费观看高清直播| 国产精品高清免费网站| 亚洲人成电影网站色| 精品日韩亚洲AV无码| 亚洲熟伦熟女新五十路熟妇| 最近最新中文字幕完整版免费高清| 大地影院MV在线观看视频免费| 国产亚洲欧美日韩亚洲中文色| 亚洲欧洲日韩国产| 久久精品国产亚洲av麻| 亚洲精品无码永久在线观看| 免费毛片在线视频| 毛色毛片免费观看| 国产免费不卡v片在线观看 | www.亚洲精品| 免费高清在线爱做视频|