计算
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(cocia@163.com)
是程序员。他拥有学士学位,现在专攻
Java
相关技术,刚刚开始在计算机领域折腾。