计算 Java 日期--学习怎样创建和使用日期
概要
不管你是处理财务交易还是计划着下一步的行动,你都要知道怎样在 Java 中建立,使用和显示日期。这需要你简单的查阅一下相应类的 API 参考:一个日期可以创建 3 个相关类的对象。这篇文章告诉你你想要知道的内容。( 3,000 字)
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() 方法找到这个日期代表的毫秒数量:
1.
2. import java.util.*;
3.
4.
5. public class Now {
6. public static void main(String[] args) {
7. Date now = new Date();
8. long nowLong = now.getTime();
9. System.out.println("Value is " + nowLong);
10. }
11. }
当我运行这个程序后,我得到 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() 方法:
1.
2. DateFormat df = DateFormat.getDateInstance();
DateFormat 类在 java.text 包中可以找到。
转换成字符串
你可以使用 format() 方法转换 Date 对象为一个字符串。下面的示例程序说明了这个问题:
1.
2. import java.util.*;
3. import java.text.*;
4.
5. public class NowString {
6. public static void main(String[] args) {
7. Date now = new Date();
8. DateFormat df = DateFormat.getDateInstance();
9. String s = df.format(now);
10. System.out.println("Today is " + s);
11. }
12. }
在上面的代码中,展示了没有参数,使用缺省格式的 getDateInstance() 方法。 Java 还提供了几个选择日期格式,你可以通过使用重载的 getDateInstance(int style) 获得。出于方便的原因, DateFormat 提供了几种预置的常量,你可以使用这些常量参数。下面是几个 SHORT, MEDIUM, LONG, 和 FULL 类型的示例:
1.
2. import java.util.*;
3. import java.text.*;
4.
5. public class StyleDemo {
6. public static void main(String[] args) {
7. Date now = new Date();
8.
9. DateFormat df = DateFormat.getDateInstance();
10. DateFormat df1 = DateFormat.getDateInstance(DateFormat.SHORT);
11. DateFormat df2 = DateFormat.getDateInstance(DateFormat.MEDIUM);
12. DateFormat df3 = DateFormat.getDateInstance(DateFormat.LONG);
13. DateFormat df4 = DateFormat.getDateInstance(DateFormat.FULL);
14. String s = df.format(now);
15. String s1 = df1.format(now);
16. String s2 = df2.format(now);
17. String s3 = df3.format(now);
18. String s4 = df4.format(now);
19.
20. System.out.println("(Default) Today is " + s);
21. System.out.println("(SHORT) Today is " + s1);
22. System.out.println("(MEDIUM) Today is " + s2);
23. System.out.println("(LONG) Today is " + s3);
24. System.out.println("(FULL) Today is " + s4);
25. }
26. }
程序输出如下:
(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 对象:
1.
2. import java.util.*;
3. import java.text.*;
4.
5. public class ParseExample {
6. public static void main(String[] args) {
7. String ds = "November 1, 2000";
8. DateFormat df = DateFormat.getDateInstance();
9. try {
10. Date d = df.parse(ds);
11. }
12. catch(ParseException e) {
13. System.out.println("Unable to parse " + ds);
14. }
15. }
16. }
在创建一个任意的日期时 parse() 方法很有用。我将通过另一种方法创建一个任意得日期。同时,你将看到怎样进行基本日期计算,例如计算 90 天后的另一天。你可以使用 GregorianCalendar 类来完成这个任务。
GregorianCalendar 类
创建一个代表任意日期的一个途径使用 GregorianCalendar 类的构造函数,它包含在 java.util 包中:
1.
2. GregorianCalendar (int year, int month, int date)
注意月份的表示,一月是 0 ,二月是 1 ,以此类推,是 12 月是 11 。因为大多数人习惯于使用单词而不是使用数字来表示月份,这样程序也许更易读,父类 Calendar 使用常量来表示月份: JANUARY, FEBRUARY, 等等。所以,创建 Wilbur 和 Orville 制造第一架动力飞机的日期( December 17, 1903 ),你可以使用:
1.
2. GregorianCalendar firstFlight = new GregorianCalendar(1903, Calendar.DECEMBER, 17);
出于清楚的考虑,你应该使用前面的形式。但是,你也应该学习怎样阅读下面的短格式。下面的例子同样表示 December 17,1903 (记住,在短格式中, 11 表示 December )
1.
2. GregorianCalendar firstFlight = new GregorianCalendar(1903, 11, 17);
在上一节中,你学习了转换 Date 对象到字符串。这里,你可以做同样的事情;但是首先,你需要将 GregorianCalendar 对象转换到 Date 。要做到这一点,你可以使用 getTime() 方法,从它得父类 Calendar 继承而来。 GetTime() 方法返回 GregorianCalendar 相应的 Date 对象。你能够创建 GregorianCalendar 对象,转换到 Date 对象,得到和输出相应的字符串这样一个过程。下面是例子:
1.
2. import java.util.*;
3. import java.text.*;
4.
5. public class Flight {
6.
7. public static void main(String[] args) {
8. GregorianCalendar firstFlight = new GregorianCalendar(1903, Calendar.DECEMBER, 17);
9. Date d = firstFlight.getTime();
10. DateFormat df = DateFormat.getDateInstance();
11. String s = df.format(d);
12. System.out.println("First flight was " + s);
13. }
14. }
有时候创建一个代表当前时刻的 GregorianCalendar 类的实例是很有用的。你可以简单的使用没有参数的 GregorianCalendar 构造函数,象这样:
1.
2. GregorianCalendar thisday = new GregorianCalendar();
一个输出今天日期的例子程序,使用 GregorianCalendar 对象:
1.
2. import java.util.*;
3. import java.text.*;
4.
5. class Today {
6. public static void main(String[] args) {
7. GregorianCalendar thisday = new GregorianCalendar();
8. Date d = thisday.getTime();
9. DateFormat df = DateFormat.getDateInstance();
10. String s = df.format(d);
11. System.out.println("Today is " + s);
12. }
13. }
注意到, Date() 构造函数和 GregorianCalendar() 构造函数很类似:都创建一个对象,条件简单,代表今天。
日期处理
GregorianCalendar 类提供处理日期的方法。一个有用的方法是 add(). 使用 add() 方法,你能够增加象年,月数,天数到日期对象中。要使用 add() 方法,你必须提供要增加的字段,要增加的数量。一些有用的字段是 DATE, MONTH, YEAR, 和 WEEK_OF_YEAR 。下面的程序使用 add() 方法计算未来 80 天的一个日期。在 Jules 的 < 环球 80 天 > 是一个重要的数字,使用这个程序可以计算 Phileas Fogg 从出发的那一天 1872 年 10 月 2 日后 80 天的日期:
1.
2. import java.util.*;
3. import java.text.*;
4.
5. public class World {
6. public static void main(String[] args) {
7. GregorianCalendar worldTour = new GregorianCalendar(1872, Calendar.OCTOBER, 2);
8. worldTour.add(GregorianCalendar.DATE, 80);
9. Date d = worldTour.getTime();
10. DateFormat df = DateFormat.getDateInstance();
11. String s = df.format(d);
12. System.out.println("80 day trip will end " + s);
13. }
14. }
这个例子是想象的,但在一个日期上增加天数是一个普遍的操作:影碟可以租 3 天,图书馆可以借书 21 天,商店经常需要将购买的物品在 30 天内卖出。下面的程序演示了使用年计算:
1.
2. import java.util.*;
3. import java.text.*;
4.
5. public class Mortgage {
6. public static void main(String[] args) {
7. GregorianCalendar mortgage = new GregorianCalendar(1997, Calendar.MAY, 18);
8. mortgage.add(Calendar.YEAR, 15);
9. Date d = mortgage.getTime();
10. DateFormat df = DateFormat.getDateInstance();
11. String s = df.format(d);
12. System.out.println("15 year mortgage amortized on " + s); }
13. }
add() 一个重要的副作用是它改变的原来的日期。有时候,拥有原始日期和修改后的日期很重要。不幸的是,你不能简单的创建一个 GregorianCalendar 对象,设置它和原来的相等( equal )。原因是两个变量指向同一个 Date() 对象地址。如果 Date 对象改变,两个变量就指向改变后的日期对象。代替这种做法,应该创建一个新对象。下面的程序示范了这种做法:
1.
2. import java.util.*;
3. import java.text.*;
4.
5. public class ThreeDates {
6. public static void main(String[] args) {
7. GregorianCalendar gc1 = new GregorianCalendar(2000, Calendar.JANUARY, 1);
8. GregorianCalendar gc2 = gc1;
9. GregorianCalendar gc3 = new GregorianCalendar(2000, Calendar.JANUARY, 1);
10. //Three dates all equal to January 1, 2000
11.
12. gc1.add(Calendar.YEAR, 1);
13. file://gc1 and gc2 are changed
14.
15. DateFormat df = DateFormat.getDateInstance();
16.
17. Date d1 = gc1.getTime();
18. Date d2 = gc2.getTime();
19. Date d3 = gc3.getTime();
20.
21. String s1 = df.format(d1);
22. String s2 = df.format(d2);
23. String s3 = df.format(d3);
24.
25. System.out.println("gc1 is " + s1);
26. System.out.println("gc2 is " + s2);
27. System.out.println("gc3 is " + s3);
28. }
29. }
程序运行后, 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 类显示在下面:
1.
2. import java.util.*;
3. import java.text.*;
4.
5. public class ReviewDates {
6. private GregorianCalendar firstDay, oneDay, oneWeek, oneMonth, oneQuarter, oneYear;
7. final int dateArraySize = 6;
8.
9. ReviewDates(GregorianCalendar gcDate) {
10. int year = gcDate.get(GregorianCalendar.YEAR);
11. int month = gcDate.get(GregorianCalendar.MONTH);
12. int date = gcDate.get(GregorianCalendar.DATE);
13.
14. firstDay = new GregorianCalendar(year, month, date);
15. oneDay = new GregorianCalendar(year, month, date);
16. oneWeek = new GregorianCalendar(year, month, date);
17. oneMonth = new GregorianCalendar(year, month, date);
18. oneQuarter = new GregorianCalendar(year, month, date);
19. oneYear = new GregorianCalendar(year, month, date);
20.
21. oneDay.add(GregorianCalendar.DATE, 1);
22. oneWeek.add(GregorianCalendar.DATE, 7);
23. oneMonth.add(GregorianCalendar.MONTH, 1);
24. oneQuarter.add(GregorianCalendar.MONTH, 3);
25. oneYear.add(GregorianCalendar.YEAR, 1);
26. }
27.
28. ReviewDates() {
29. this(new GregorianCalendar());
30. }
31.
32. public void listDates() {
33. DateFormat df = DateFormat.getDateInstance(DateFormat.LONG);
34. Date startDate = firstDay.getTime();
35. Date date1 = oneDay.getTime();
36. Date date2 = oneWeek.getTime();
37. Date date3 = oneMonth.getTime();
38. Date date4 = oneQuarter.getTime();
39. Date date5 = oneYear.getTime();
40.
41. String ss = df.format(startDate);
42. String ss1 = df.format(date1);
43. String ss2 = df.format(date2);
44. String ss3 = df.format(date3);
45. String ss4 = df.format(date4);
46. String ss5 = df.format(date5);
47.
48. System.out.println("Start date is " + ss);
49. System.out.println("Following review dates are:");
50. System.out.println(ss1);
51. System.out.println(ss2);
52. System.out.println(ss3);
53. System.out.println(ss4);
54. System.out.println(ss5);
55. System.out.println();
56. }
57.
58. public GregorianCalendar[] getDates() {
59. GregorianCalendar[] memoryDates = new GregorianCalendar[dateArraySize];
60. memoryDates[0] = firstDay;
61. memoryDates[1] = oneDay;
62. memoryDates[2] = oneWeek;
63. memoryDates[3] = oneMonth;
64. memoryDates[4] = oneQuarter;
65. memoryDates[5] = oneYear;
66. return memoryDates;
67. }
68.
69. public GregorianCalendar getFirstDay() {
70. return this.firstDay;
71. }
72.
73. public GregorianCalendar getOneDay() {
74. return this.oneDay;
75. }
76.
77. public GregorianCalendar getOneWeek() {
78. return this.oneWeek;
79. }
80.
81. public GregorianCalendar getOneMonth() {
82. return this.oneMonth;
83. }
84.
85. public GregorianCalendar getOneQuarter() {
86. return this.oneQuarter;
87. }
88.
89. public GregorianCalendar getOneYear() {
90. return this.oneYear;
91. }
92. }
下面是使用 ReviewDates 类列出复习日期的例子程序:
1.
2. import java.util.*;
3.
4. public class ShowDates {
5. public static void main(String[] args) {
6. ReviewDates rd = new ReviewDates();
7. rd.listDates();
8.
9. GregorianCalendar gc = new GregorianCalendar(2001, Calendar.JANUARY, 15);
10. ReviewDates jan15 = new ReviewDates(gc);
11. jan15.listDates();
12. }
13. }
总结
这篇文章介绍了关于日期处理的 3 个重要的类: Date,DateFormat,GregorianCalendar. 这些类让你创建日期,转换成字符串,和计算日期基本元素。处理 Java 中的日期问题,这篇文章只是冰山一角。可是,我在这里介绍的类和方法不仅仅是你学习高级技术的跳板,这些类和方法本身就可以处理很多通常的日期相关的任务
关于作者
Robert Nielsen 是 SCJP 。他拥有硕士学位,专攻计算机教育,并且在计算机领域执教多年。他也在各样的杂志上发表过很多计算机相关的文章。
关于译者
Cocia Lin([email protected]) 是程序员。他拥有学士学位,现在专攻 Java 相关技术,刚刚开始在计算机领域折腾。