学习Java中的日期和时间处理及Java日历小程序的编写
java 在 java.util 包中提供了 date 类,这个类封装了当前的日期和时间。
date 类支持两种构造函数。第一个构造函数初始化对象的当前日期和时间。
date( )
下面的构造函数接收一个参数等于自1970年1月1日午夜起已经过的毫秒数
date(long millisec)
一旦有一个可用的日期对象,可以调用以下任何一种支持的方法使用时间:
sn | 方法和描述 |
---|---|
1 | boolean after(date date) 如果调用date对象包含或晚于指定的日期则返回true,否则,返回false。 |
2 | boolean before(date date) 如果调用date对象包含或早于日期指定的日期返回true,否则,返回false。 |
3 | object clone( ) 重复调用date对象。 |
4 | int compareto(date date) 调用对象的值与日期比较。如果这两个值相等返回0。如果调用对象是早于日期返回一个负值。如果调用对象迟于日期返回正值。 |
5 | int compareto(object obj) 以相同的compareto(date)操作 如果obj是一个类日期。否则,它会抛出一个classcastexception。 |
6 | boolean equals(object date) 如果调用date对象包含相同的时间及日期指定日期则返回true,否则,返回false。 |
7 | long gettime( ) 返回自1970年1月1日起已经过的毫秒数。 |
8 | int hashcode( ) 返回调用对象的哈希代码。 |
9 | void settime(long time) 设置所指定的时间,这表示从1970年1月1日从午夜的时间和日期以毫秒为单位经过的时间。 |
10 | string tostring( ) 调用date对象转换为字符串,并返回结果。 |
获取当前日期和时间
在 java 中容易得到当前的日期和时间。可以使用一个简单的 date 对象的 tostring() 方法,如下所示打印当前日期和时间:
import java.util.date; public class datedemo { public static void main(string args[]) { // instantiate a date object date date = new date(); // display time and date using tostring() system.out.println(date.tostring()); } }
这将产生以下结果:
mon may 04 09:51:52 cdt 2009
日期比较
有以下三种方式来比较两个日期:
- 可以使用 gettime() 来获得自1970年1月1日午夜十二时起已经过的毫秒数,然后比较两个对象的值。
- 可以使用 before( ), after( ), 和 equals( ) 方法,由于12日在18日前,例如, new date(99, 2, 12).before(new date (99, 2, 18)) 返回值为 true。
- 可以使用 compareto() 方法,这是由 comparable 接口定义,由 date 实现。
使用 simpledateformat 格式化日期
simpledateformat 是一个具体的类,以本地方式用于格式化和转换日期。simpledateformat 允许选择用户定义的模式为日期时间格式。例如:
import java.util.*; import java.text.*; public class datedemo { public static void main(string args[]) { date dnow = new date( ); simpledateformat ft = new simpledateformat ("e yyyy.mm.dd 'at' hh:mm:ss a zzz"); system.out.println("current date: " + ft.format(dnow)); } }
这将产生以下结果:
current date: sun 2004.07.18 at 04:14:09 pm pdt
简单的 dateformat 格式代码
要指定时间格式,使用时间模式字符串。在这个模式中,所有的 ascii 字母被保留为模式字母,其定义如下:
字符 | 描述 | 例子 |
---|---|---|
g | 时代指示器 | ad |
y | 四位数年份 | 2001 |
m | 年中的月份 | july or 07 |
d | 月份中日期 | 10 |
h | 时间 a.m./p.m.(1~12) | 12 |
h | 天中的小时 (0~23) | 22 |
m | 小时中的分钟 | 30 |
s | 分钟中的秒钟 | 55 |
s | 毫秒 | 234 |
e | 星期中的天 | tuesday |
d | 年中的天 | 360 |
f | 月中星期中的天 | 2 (second wed. in july) |
w | 年中的星期 | 40 |
w | 月中的星期 | 1 |
a | a.m./p.m. 标记 | pm |
k | 天中的小时(1~24) | 24 |
k | 小时a.m./p.m. (0~11) | 10 |
z | 时区 | 东部标准时间 |
' | 脱离文本 | 分隔符 |
" | 单引号 | ` |
用 printf 格式化日期
日期和时间格式用 printf 方法可以非常轻松地做到。可以使用两个字母的格式,从 t 开始并在下面给出的表格中的其中一个字母结束。例如:
import java.util.date; public class datedemo { public static void main(string args[]) { // instantiate a date object date date = new date(); // display time and date using tostring() string str = string.format("current date/time : %tc", date ); system.out.printf(str); } }
这将产生以下结果:
current date/time : sat dec 15 16:37:57 mst 2012
如果提供日期多次格式化是一种不好的做法。一个格式字符串可以指示要格式化的参数的索引。
索引必须紧跟在 % 之后,并必须由 $ 终止。例如:
import java.util.date; public class datedemo { public static void main(string args[]) { // instantiate a date object date date = new date(); // display time and date using tostring() system.out.printf("%1$s %2$tb %2$td, %2$ty", "due date:", date); } }
这将产生以下结果:
due date: february 09, 2004
或者,也可以使用 < 标志。则表示相同的参数,根据前述格式规范,应再次使用。例如:
import java.util.date; public class datedemo { public static void main(string args[]) { // instantiate a date object date date = new date(); // display formatted date system.out.printf("%s %tb %<te, %<ty", "due date:", date); } }
这将产生以下结果:
due date: february 09, 2004
日期和时间转换字符
字符 | 描述 | 例子 |
---|---|---|
c | 完整的日期和时间 | mon may 04 09:51:52 cdt 2009 |
f | iso 8601 日期 | 2004-02-09 |
d | u.s. 格式时间 (月/日/年) | 02/09/2004 |
t | 24-时制 | 18:05:19 |
r | 12-时制 | 06:05:19 pm |
r | 24-时制,无秒 | 18:05 |
y | 四位数年份 (用前行零列) | 2004 |
y | 年份的后两位数(用前行零列) | 04 |
c | 年份的前两位(用前行零列) | 20 |
b | 完整月份名称 | february |
b | 缩写月份名称 | feb |
m | 两位数月份 (用前行零列) | 02 |
d | 两位数日期 (用前行零列) | 03 |
e | 两位数日期(无前行零列) | 9 |
a | 完整星期名称 | monday |
a | 缩写星期名称 | mon |
j | 年中的三位数天数(用前行零列) | 069 |
h | 两位数小时(用前行零列), 00 和 23之间 | 18 |
k | 两位数小时(无前行零列), 0 和 23 之间 | 18 |
i | 两位数小时 (用前行零列), 01和12之间 | 06 |
l | 两位数小时 (无前行零列), 1 和12之间 | 6 |
m | 两位数分钟 (用前行零列) | 05 |
s | 两位数秒钟(用前行零列) | 19 |
l | 三位数毫秒(用前行零列) | 047 |
n | 九位数纳秒 (用前行零列) | 047000000 |
p | 大写上下午标记 | pm |
p | 小写上下午标记 | pm |
z | rfc 822 从gmt数字抵消 | -0800 |
z | 时区 | pst |
s | 从 1970-01-01 00:00:00 的秒数gmt | 1078884319 |
q | 从 1970-01-01 00:00:00 的毫秒数gmt | 1078884319047 |
有相关的日期和时间等有用的类。欲了解更多详细信息,可以参考 java 标准文档。
字符串转换日期
simpledateformat 类有一些额外的方法,如 parse(),它试图根据存储在给定 simpledateformat 的对象的格式来转换字符串。例如:
import java.util.*; import java.text.*; public class datedemo { public static void main(string args[]) { simpledateformat ft = new simpledateformat ("yyyy-mm-dd"); string input = args.length == 0 ? "1818-11-11" : args[0]; system.out.print(input + " parses as "); date t; try { t = ft.parse(input); system.out.println(t); } catch (parseexception e) { system.out.println("unparseable using " + ft); } } }
上述程序的运行示例将产生以下结果:
$ java datedemo 1818-11-11 parses as wed nov 11 00:00:00 gmt 1818 $ java datedemo 2007-12-01 2007-12-01 parses as sat dec 01 00:00:00 gmt 2007
休眠一段时间
你可以进行任何期间的时间休眠,从一毫秒直到你的电脑的整个生命周期。例如,下面的程序会休眠10秒:
import java.util.*; public class sleepdemo { public static void main(string args[]) { try { system.out.println(new date( ) + "\n"); thread.sleep(5*60*10); system.out.println(new date( ) + "\n"); } catch (exception e) { system.out.println("got an exception!"); } } }
这将产生以下结果:
sun may 03 18:04:41 gmt 2009 sun may 03 18:04:51 gmt 2009
测量执行时间
有时候,可能需要测量的时间点以毫秒为单位。因此,让我们再一次重新写上面的例子:
import java.util.*; public class diffdemo { public static void main(string args[]) { try { long start = system.currenttimemillis( ); system.out.println(new date( ) + "\n"); thread.sleep(5*60*10); system.out.println(new date( ) + "\n"); long end = system.currenttimemillis( ); long diff = end - start; system.out.println("difference is : " + diff); } catch (exception e) { system.out.println("got an exception!"); } } }
这将产生以下结果:
sun may 03 18:16:51 gmt 2009 sun may 03 18:16:57 gmt 2009 difference is : 5993
gregoriancalendar 类
gregoriancalendar 是一个 calendar 类具体的实现,即你所熟悉的对正常 gregorian 公历的实现。本教程中不讨论 calendar 类,可以看看标准 java 文档。
calendar 的 getinstance() 方法返回与当前日期和时间默认语言环境和时区初始化的一个 gregoriancalendar。 gregoriancalendar 中定义了两个字段:ad 和 bc。这些代表在公历中定义的两个时代。
也有几个构造函数的 gregoriancalendar 对象:
sn | 构造函数描述 |
---|---|
1 | gregoriancalendar() 在默认时区与默认语言环境使用当前时间构造默认的gregoriancalendar。 |
2 | gregoriancalendar(int year, int month, int date) 在默认时区设置默认的语言环境用给定的日期构造一个gregoriancalendar |
3 | gregoriancalendar(int year, int month, int date, int hour, int minute) 用给定的日期和时间设置为与默认语言环境的默认时区构造一个gregoriancalendar。 |
4 | gregoriancalendar(int year, int month, int date, int hour, int minute, int second) 用给定的日期和时间设置为与默认语言环境的默认时区构造一个 gregoriancalendar |
5 | gregoriancalendar(locale alocale) 基于当前时间与给定语言环境的默认时区构建一个gregoriancalendar。 |
6 | gregoriancalendar(timezone zone) 基于当前时间,使用默认的语言环境在给定的时区构建一个gregoriancalendar。 |
7 | gregoriancalendar(timezone zone, locale alocale) 基于当前时间与给定语言环境的给定时区构建一个gregoriancalendar。 |
这里是由 gregoriancalendar 类提供的一些有用的方法的列表:
sn | 方法和描述 |
---|---|
1 | void add(int field, int amount) 基于日历的规则,以给定的时间字段添加指定(有符号的)时间量。 |
2 | protected void computefields() 将utc转换为毫秒时间字段值. |
3 | protected void computetime() 覆盖日历转换时间域值为utc的毫秒. |
4 | boolean equals(object obj) 这个gregoriancalendar与一个对象引用比较. |
5 | int get(int field) 获取给定时间域的值. |
6 | int getactualmaximum(int field) 返回该字段可能的最大值,给定到当前的日期. |
7 | int getactualminimum(int field) 返回该字段可能具有的最小值,给定当前的日期. |
8 | int getgreatestminimum(int field) 对于给定的字段中返回高最低值(如果有变化). |
9 | date getgregorianchange() 获取公历更改日期. |
10 | int getleastmaximum(int field) 对于给定的字段返回最低的最大值,如果有变化. |
11 | int getmaximum(int field) 返回给定字段中的最大值. |
12 | date gettime() 获取日历的当前时间. |
13 | long gettimeinmillis() 获取日历的当前时间长. |
14 | timezone gettimezone() 获取时区. |
15 | int getminimum(int field) 返回给定字段中的最小值. |
16 | int hashcode() 重写hashcode. |
17 | boolean isleapyear(int year) 确定给定年份是闰年. |
18 | void roll(int field, boolean up) 加上或减去(上/下)的时间在给定的时间字段一个单元,不更改更大的字段. |
19 | void set(int field, int value) 设置时间字段与给定值. |
20 | void set(int year, int month, int date) 设置为年,月,日的值. |
21 | void set(int year, int month, int date, int hour, int minute) 设置为年,月,日,小时和分钟值. |
22 | void set(int year, int month, int date, int hour, int minute, int second) 设置为字段的年,月,日,小时,分钟和秒的值. |
23 | void setgregorianchange(date date) 设置gregoriancalendar更改日期. |
24 | void settime(date date) 设置日历的当前时间与给定日期. |
25 | void settimeinmillis(long millis) 从给定long值设置日历的当前时间. |
26 | void settimezone(timezone value) 将时区设置与给定的时区值. |
27 | string tostring() 返回此日历的字符串表示形式. |
示例
import java.util.*; public class gregoriancalendardemo { public static void main(string args[]) { string months[] = { "jan", "feb", "mar", "apr", "may", "jun", "jul", "aug", "sep", "oct", "nov", "dec"}; int year; // create a gregorian calendar initialized // with the current date and time in the // default locale and timezone. gregoriancalendar gcalendar = new gregoriancalendar(); // display current time and date information. system.out.print("date: "); system.out.print(months[gcalendar.get(calendar.month)]); system.out.print(" " + gcalendar.get(calendar.date) + " "); system.out.println(year = gcalendar.get(calendar.year)); system.out.print("time: "); system.out.print(gcalendar.get(calendar.hour) + ":"); system.out.print(gcalendar.get(calendar.minute) + ":"); system.out.println(gcalendar.get(calendar.second)); // test if the current year is a leap year if(gcalendar.isleapyear(year)) { system.out.println("the current year is a leap year"); } else { system.out.println("the current year is not a leap year"); } } }
这将产生以下结果:
date: apr 22 2009 time: 11:25:27 the current year is not a leap year
日历小程序
下面我们来看一个日历小程序。这里用传统的mvc结构,设计3个类:calendarviewer、calendarcontrolller、calendarmodel。
calendarviewer.java主要处理ui,沿用了已有代码,整理之并抽出业务逻辑,使其专注于显示层处理。
calendarviewer.java
public class calendarviewer extends jwindow implements actionlistener { jpanel calendarympanel = null; jbutton leftbutton = new jbutton("<<"); jbutton rightbutton = new jbutton(">>"); label yearlabel = new label(); label monthlabel = new label(); label passeddayslabel = new label(); jpanel calendarwdpanel = null;// 是caledar_week和calendar_days的总包容体 jpanel calendarweekpanel = null;// 针对周列的布局 jpanel calendardayspanel = null;// 针对日期列的布局 jpanel calendarexitpanel = null; jbutton quitbutton = new jbutton("关闭"); border emptyborder = borderfactory.createemptyborder(); calendarcontroller ccontroller = new calendarcontroller(); public calendarviewer() { super(); buildui(); } public void buildui() { buildtoppanel(); buildcenterpanel(); buildbottompanel(); setlayout(new borderlayout()); 。。。。。。 } private void buildtoppanel() {。。。。。。} private void buildcenterpanel() {。。。。。。} private void buildbottompanel() {。。。。。。} public jpanel updatedayspanel() {。。。。。。} public void updatepasseddayslabel() {。。。。。。} public void actionperformed(actionevent e) {。。。。。。} public static void main(string[] args) { swingutilities.invokelater(new runnable() { public void run() { new calendarviewer(); } }); } }
ui构造主要分3块,对应图上中下3个panel。
- buildtoppanel();
- buildcenterpanel();
- buildbottompanel();
事件监听的处理由下面方法完成。
- actionperformed(actionevent e);
基于事件的ui更新由以下两个方法完成。
updatedayspanel(); updatepasseddayslabel(); calendarcontroller.java主要处理具体的业务逻辑,而所使用的一些与具体应用无关的日历算法逻辑则交给calendarmodel.java。 calendarmodel.java public class calendarmodel { private int daytab[][] = { { 0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 }, { 0, 31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 } }; public boolean isleapyear(int year) { return ((year % 4 == 0 && year % 100 != 0) || year % 400 == 0); } public int dayofyear(int day, int month, int year) { int leap = isleapyear(year) ? 1 : 0; for (int i = 1; i < month; i++) day += daytab[leap][i]; return day; } public int daysofmonth(int month, int year) { int leap = isleapyear(year) ? 1 : 0; return daytab[leap][month]; } public int dayofweek(int day, int month, int year) { if (month == 1) { month = 13; year--; } if (month == 2) { month = 14; year--; } return (day + 2 * month + 3 * (month + 1) / 5 + year + year / 4 - year / 100 + year / 400) % 7 + 1; } }
建立一个二维数组,分别表示闰年与非闰年的每月天数。主要方法有:
- boolean isleapyear(int year);判断闰年
- dayofyear(int day, int month, int year);计算所提供日期为当前year的第几天
- daysofmonth(int month, int year);返回当前月份的天数
- dayofweek(int day, int month, int year); 计算某年某月某日是星期几,这里使用了基姆拉尔森计算公式。
基姆拉尔森计算公式
- w= (d+2*m+3*(m+1)/5+y+y/4-y/100+y/400) mod 7
- d 天
- m 月
- y 年
- 1月2月换算为去年的13 14月计算
- w=0是星期一,依次类推。