Java中date处理
在
JDK1.0
中,Date
类是唯一的一个代表时间的类,但是由于Date
类不便于实现国际化,所以从JDK1.1
版本开始,推荐使用Calendar
类进行时间和日期处理。这里简单介绍一下Date
类的使用
1 Date类型
1.1 使用Date类代表当前系统时间
Date d = new Date();
System.out.println(d);
使用Date
类的默认构造方法创建出的对象就代表当前时间,由于Date
类覆盖了toString
方法,所以可以直接输出Date
类型的对象,显示的结果如下:
Sun Mar 08 16:35:58 CST 2009
在该格式中,Sun
代表Sunday(周日)
,Mar
代表March(三月)
,08
代表8
号,CST
代表China Standard Time
(中国标准时间,也就是北京时间(东八区))。
1.2 使用Date类代表指定的时间(已过时)
Date d1 = new Date(2009-1900,3-1,9);
System.out.println(d1);
使用带参数的构造方法,可以构造指定日期的Date
类对象,Date
类中年份的参数应该是实际需要代表的年份减去1900
不然的话年份就是向后推1900年
,月份根据实际需要代表的月份减去1以后的值。例如上面的示例代码代表就是2009年3月9号
实际代表具体的年月日时分秒的日期对象,和这个类似。
1.3 获得Date对象中的信息
Date d2 = new Date();
年份
int year = d2.getYear() + 1900;
月份
int month = d2.getMonth() + 1;
日期
int date = d2.getDate();
小时
int hour = d2.getHours();
分钟
int minute = d2.getMinutes();
秒
int second = d2.getSeconds();
星期几
int day = d2.getDay();
System.out.println("年份:" + year);
System.out.println("月份:" + month);
System.out.println("日期:" + date);
System.out.println("小时:" + hour);
System.out.println("分钟:" + minute);
System.out.println("秒:" + second);
System.out.println("星期:" + day);
使用Date
类中对应的get
方法,可以获得Date
类对象中相关的信息,需要注意的是使用getYear
获得是Date
对象中年份减去1900
以后的值,所以需要显示对应的年份则需要在返回值的基础上加上1900
,月份类似。在Date
类中还提供了getDay
方法,用于获得Date
对象代表的时间是星期几,Date
类规定周日是0,周一是1,周二是2
,后续的依次类推。
1.4 Date对象和相对时间之间的互转
Date d3 = new Date(2009-1900,3-1,10);
long time = 1290876532190L;
将Date类的对象转换为相对时间
long t = d3.getTime();
System.out.println(t);
将相对时间转换为Date类的对象
Date d4 = new Date(time);
System.out.println(d4);
使用Date
对象中的getTime
方法,可以将Date
类的对象转换为相对时间,使用Date
类的构造方法,可以将相对时间转换为Date
类的对象。经过转换以后,既方便了时间的计算,也使时间显示比较直观了。
2 Calendar类
从JDK1.1
版本开始,在处理日期和时间时,系统推荐使用Calendar
类进行实现。在设计上,Calendar
类的功能要比Date
类强大很多,而且在实现方式上也比Date
类要复杂一些,下面就介绍一下Calendar
类的使用。
Calendar
类是一个抽象类,在实际使用时实现特定的子类的对象,创建对象的过程对程序员来说是透明的,只需要使用getInstance
方法创建即可。
2.1 使用Calendar类代表当前时间
Calendar c = Calendar.getInstance();
由于Calendar
类是抽象类,且Calendar
类的构造方法是protected
的,所以无法使用Calendar
类的构造方法来创建对象,API
中提供了getInstance
方法用来创建对象。
使用该方法获得的Calendar
对象就代表当前系统时间
,由于Calendar
类toString
实现的没有Date
类那么直观,所以直接输出Calendar
类的对象意义不大。
2.2 使用Calendar类代表指定的时间
Calendar c1 = Calendar.getInstance();
c1.set(2009, 3-1, 9);
使用Calendar
类代表特定的时间,需要首先创建一个Calendar
的对象,然后再设定该对象中的年月日参数来完成。
set
方法的声明为:
public final void set(int year,int month,int date)
以上示例代码设置的时间为2009
年3
月9
日,其参数的结构和Date
类不一样。Calendar
类中年份的数值直接书写,月份的值为实际的月份值减1
,日期的值就是实际的日期值。
如果只设定某个字段,例如日期的值,则可以使用如下set方法:
public void set(int field,int value)
在该方法中,参数field
代表要设置的字段的类型,常见类型如下:
Calendar.YEAR——年份
Calendar.MONTH——月份
Calendar.DATE——日期
Calendar.DAY_OF_MONTH——日期,和上面的字段完全相同
Calendar.HOUR——12小时制的小时数
Calendar.HOUR_OF_DAY——24小时制的小时数
Calendar.MINUTE——分钟
Calendar.SECOND——秒
Calendar.DAY_OF_WEEK——星期几
后续的参数value
代表,设置成的值。例如:
c1.set(Calendar.DATE,10);
该代码的作用是将c1
对象代表的时间中日期设置为10
号,其它所有的数值会被重新计算,例如星期几以及对应的相对时间数值等。
另外要注意的一点是,Calendar
为了性能原因对 set()
方法采取延缓计算
的方法。在 JavaDoc
中有下面的例子来说明这个问题:
Calendar cal1 = Calendar.getInstance();
cal1.set(2000, 7, 31, 0, 0 , 0); //2000-8-31
cal1.set(Calendar.MONTH, Calendar.SEPTEMBER); //应该是 2000-9-31,也就是 2000-10-1
cal1.set(Calendar.DAY_OF_MONTH, 30); //如果 Calendar 转化到 2000-10-1,那么现在的结果就该是 2000-10-30
System.out.println(new SimpleDateFormat("yyyyMMdd HHmmss").format(cal1.getTime())); //输出的是2000-9-30,说明 Calendar 不是马上就刷新其内部的记录
在Calendar
的方法中,get()
和 add()
会让 Calendar
立刻刷新。Set()
的这个特性会给我们的开发带来一些意想不到的结果。我们后面会看到这个问题
2.3 获得Calendar类中的信息
Calendar c2 = Calendar.getInstance();
年份
int year = c2.get(Calendar.YEAR);
月份
int month = c2.get(Calendar.MONTH) + 1;
日期
int date = c2.get(Calendar.DATE);
小时
int hour = c2.get(Calendar.HOUR_OF_DAY);
分钟
int minute = c2.get(Calendar.MINUTE);
秒
int second = c2.get(Calendar.SECOND);
星期几
int day = c2.get(Calendar.DAY_OF_WEEK);
使用Calendar
类中的get
方法可以获得Calendar
对象中对应的信息,get
方法的声明如下:
public int get(int field)
其中参数field
代表需要获得的字段的值,字段说明和上面的set
方法保持一致。需要说明的是,获得的月份为实际的月份值减1
,获得的星期的值和Date
类不一样。在Calendar
类中,周日是1,周一是2,周二是3
,依次类推
2.4 其它方法说明
其实Calendar
类中还提供了很多其它有用的方法,下面简单的介绍几个常见方法的使用。
2.4.1 add和roll
public abstract void add(int field,int amount)
该方法的作用是在Calendar
对象中的某个字段上增加或减少一定的数值,增加是amount
的值为正,减少时amount
的值为负。
例如在计算一下当前时间100
天以后的日期,代码如下:
Calendar c3 = Calendar.getInstance();
c3.add(Calendar.DATE, 100);
int year1 = c3.get(Calendar.YEAR);
月份
int month1 = c3.get(Calendar.MONTH) + 1;
日期
int date1 = c3.get(Calendar.DATE);
System.out.println(year1 + "年" + month1 + "月" + date1 + "日");
这里add
方法是指在c3
对象的Calendar.DATE
,也就是日期字段上增加100
,类内部会重新计算该日期对象中其它各字段的值,从而获得100
天以后的日期
add() 有两条规则:
- 当被修改的字段超出它可以的范围时,那么比它大的字段会自动修正
- 如果比它小的字段是不可变的(由
Calendar
的实现类决定),那么该小字段会修正到变化最小的值。
Roll()
的规则只有一条:
当被修改的字段超出它可以的范围时,那么比它大的字段不会被修正。如:
Calendar cal1 = Calendar.getInstance();
cal1.set(1999, 5, 6, 0, 0, 0); //1999-6-6, 周日
cal1.roll(Calendar.WEEK_OF_MONTH, -1); //1999-6-1, 周二
cal1.set(1999, 5, 6, 0, 0, 0); //1999-6-6, 周日
cal1.add(Calendar.WEEK_OF_MONTH, -1); //1999-5-30, 周日
WEEK_OF_MONTH
比 MONTH
字段小,所以 roll
不能修正 MONTH
字段
roll方法在本月内循环
Calendar cal= Calendar.getInstance();
cal.set(Calendar.YEAR, 2006);
cal.set(Calendar.MONTH, 8);
cal.set(Calendar.DAY_OF_MONTH, 3);
cal.roll(Calendar.DATE, -4);
date=cal.getTime();
System.out.println(df.format(date));
cal.roll(Calendar.DATE, 4);
date=cal.getTime();
System.out.println(df.format(date));
输出:
2006-09-29
2006-09-03
2.4.2 after方法
public boolean after(Object when)
该方法的作用是判断当前日期对象是否在when
对象的后面,如果在when
对象的后面则返回true
,否则返回false
。例如:
Calendar c4 = Calendar.getInstance();
c4.set(2009, 10 - 1, 10);
Calendar c5 = Calendar.getInstance();
c5.set(2010, 10 - 1, 10);
boolean b = c5.after(c4);
System.out.println(b);
在该示例代码中对象c4代表的时间是2009年10月10号,对象c5代表的时间是2010年10月10号,则对象c5代表的日期在c4代表的日期之后,所以after方法的返回值是true。
另外一个类似的方法是before
,该方法是判断当前日期对象是否位于另外一个日期对象之前。
2.4.3 getTime方法
public final Date getTime()
该方法的作用是将Calendar
类型的对象转换为对应的Date
类对象,两者代表相同的时间点。
类似的方法是setTime
,该方法的作用是将Date
对象转换为对应的Calendar
对象,该方法的声明如下:
public final void setTime(Date date)
转换的示例代码如下:
Date d = new Date();
Calendar c6 = Calendar.getInstance();
//Calendar类型的对象转换为Date对象
Date d1 = c6.getTime();
//Date类型的对象转换为Calendar对象
Calendar c7 = Calendar.getInstance();
c7.setTime(d);
2.5 Calendar对象和相对时间之间的互转
Calendar c8 = Calendar.getInstance();
long t = 1252785271098L;
//将Calendar对象转换为相对时间
long t1 = c8.getTimeInMillis();
//将相对时间转换为Calendar对象
Calendar c9 = Calendar.getInstance();
c9.setTimeInMillis(t1);
在转换时,使用Calendar
类中的getTimeInMillis
方法可以将Calendar
对象转换为相对时间。在将相对时间转换为Calendar
对象时,首先创建一个Calendar
对象,然后再使用Calendar
类的setTimeInMillis
方法设置时间即可。
2.6 应用示例
下面以两个简单的示例介绍时间和日期处理的基本使用。
2.6.1 计算两个日期之间相差的天数
例如计算2010年4月1号和2009年3月11号之间相差的天数,则可以使用时间和日期处理进行计算。
该程序实现的原理为:首先代表两个特定的时间点,这里使用Calendar
的对象进行代表,然后将两个时间点转换为对应的相对时间,求两个时间点相对时间的差值,然后除以1天的毫秒数(24小时X60分钟X60秒X1000毫秒)即可获得对应的天数。实现该示例的完整代码如下:
import java.util.*;
/**
* 计算两个日期之间相差的天数
*/
public class DateExample1 {
public static void main(String[] args) {
//设置两个日期
//日期:2009年3月11号
Calendar c1 = Calendar.getInstance();
c1.set(2009, 3 - 1, 11);
//日期:2010年4月1号
Calendar c2 = Calendar.getInstance();
c2.set(2010, 4 - 1, 1);
//转换为相对时间
long t1 = c1.getTimeInMillis();
long t2 = c2.getTimeInMillis();
//计算天数
long days = (t2 - t1)/(24 * 60 * 60 * 1000);
System.out.println(days);
}
}
2.6.2 输出当前月的月历
该示例的功能是输出当前系统时间所在月的日历,例如当前系统时间是2009年3月10日,则输出2009年3月的日历。
该程序实现的原理为:首先获得该月1号是星期几,然后获得该月的天数,最后使用流程控制实现按照日历的格式进行输出即可。即如果1号是星期一,则打印一个单位的空格,如果1号是星期二,则打印两个单位的空格,依次类推。打印完星期六的日期以后,进行换行。实现该示例的完整代码如下:
import java.util.*;
/**
* 输出当前月的日历
*/
public class DateExample2{
public static void main(String[] args){
//获得当前时间
Calendar c = Calendar.getInstance();
//设置代表的日期为1号
c.set(Calendar.DATE,1);
//获得1号是星期几
int start = c.get(Calendar.DAY_OF_WEEK);
//获得当前月的最大日期数
int maxDay = c.getActualMaximum(Calendar.DATE);
//输出标题
System.out.println("星期日 星期一 星期二 星期三 星期四 星期五 星期六");
//输出开始的空格
for(int i = 1;i < start;i++){
System.out.print(" ");
}
//输出该月中的所有日期
for(int i = 1;i <= maxDay;i++){
//输出日期数字
System.out.print(" " + i);
//输出分隔空格
System.out.print(" ");
if(i < 10){
System.out.print(' ');
}
//判断是否换行
if((start + i - 1) % 7 == 0){
System.out.println();
}
}
//换行
System.out.println();
}
}
关于时间和日期的处理就介绍这么多,更多的实现方法还需要根据具体问题进行对应的实现。
2.6.3 计算某一月份的最大天数
Calendar time=Calendar.getInstance();
time.clear();
time.set(Calendar.YEAR,year);
time.set(Calendar.MONTH,i-1);//注意,Calendar对象默认一月为0
int day=time.getActualMaximum(Calendar.DAY_OF_MONTH);//本月份的天数
注
:在使用set
方法之前,必须先clear
一下,否则很多信息会继承自系统当前时间
2.6.4 计算一年中的第几星期
计算某一天是一年中的第几星期
Calendar cal=Calendar.getInstance();
cal.set(Calendar.YEAR, 2006);
cal.set(Calendar.MONTH, 8);
cal.set(Calendar.DAY_OF_MONTH, 3);
int weekno=cal.get(Calendar.WEEK_OF_YEAR);
计算一年中的第几星期是几号
SimpleDateFormat df=new SimpleDateFormat("yyyy-MM-dd");
Calendar cal=Calendar.getInstance();
cal.set(Calendar.YEAR, 2006);
cal.set(Calendar.WEEK_OF_YEAR, 1);
cal.set(Calendar.DAY_OF_WEEK, Calendar.MONDAY);
System.out.println(df.format(cal.getTime()));
输出:
2006-01-02
3 JDK8中带的date类型
Java 8
通过发布新的Date-Time API (JSR 310)
来进一步加强对日期与时间的处理。
在旧版的 Java
中,日期时间 API
存在诸多问题,其中有:
- 非线程安全 −
java.util.Date
是非线程安全的,所有的日期类都是可变的,这是Java
日期类最大的问题之一。 - 时区处理麻烦 − 日期类并不提供国际化,没有时区支持,因此Java引入了
java.util.Calendar
和java.util.TimeZone
类,但他们同样存在上述所有的问题。
Java 8
在 java.time
包下提供了很多新的 API。以下为两个比较重要的 API:
Local
(本地) − 简化了日期时间的处理,没有时区的问题。Zoned
(时区) − 通过制定的时区处理日期时间。
3.1 简单使用
public class Java8Tester {
public static void main(String args[]){
Java8Tester java8tester = new Java8Tester();
java8tester.testLocalDateTime();
}
public void testLocalDateTime(){
// 获取当前的日期时间
LocalDateTime currentTime = LocalDateTime.now();
System.out.println("当前时间: " + currentTime);
LocalDate date1 = currentTime.toLocalDate();
System.out.println("date1: " + date1);
Month month = currentTime.getMonth();
int day = currentTime.getDayOfMonth();
int seconds = currentTime.getSecond();
System.out.println("月: " + month +", 日: " + day +", 秒: " + seconds);
LocalDateTime date2 = currentTime.withDayOfMonth(10).withYear(2012);
System.out.println("date2: " + date2);
// 12 december 2014
LocalDate date3 = LocalDate.of(2014, Month.DECEMBER, 12);
System.out.println("date3: " + date3);
// 22 小时 15 分钟
LocalTime date4 = LocalTime.of(22, 15);
System.out.println("date4: " + date4);
// 解析字符串
LocalTime date5 = LocalTime.parse("20:15:30");
System.out.println("date5: " + date5);
}
}
3.2 Date转LocalDateTime
Date date = new Date();
Instant instant = date.toInstant();
//获取时区
ZoneId zone = ZoneId.of("Asia/Shanghai") == null ?
ZoneId.systemDefault() : ZoneId.of("Asia/Shanghai");
LocalDateTime localDateTime = LocalDateTime.ofInstant(instant, zone);
System.out.println(localDateTime.format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")));
3.3 LocalDateTime转Date
LocalDateTime now = LocalDateTime.now();
ZoneId zone = ZoneId.of("Asia/Shanghai") == null ?
ZoneId.systemDefault() : ZoneId.of("Asia/Shanghai");
Instant instant = now.atZone(zone).toInstant();
Date date = Date.from(instant);
System.out.println(sdf.format(date));
//LocalDateTime 默认格式是2021-10-16T21:09:35.902
System.out.println(now.toString());
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 探究高空视频全景AR技术的实现原理
· 理解Rust引用及其生命周期标识(上)
· 浏览器原生「磁吸」效果!Anchor Positioning 锚点定位神器解析
· 没有源码,如何修改代码逻辑?
· 一个奇形怪状的面试题:Bean中的CHM要不要加volatile?
· 分享4款.NET开源、免费、实用的商城系统
· 全程不用写代码,我用AI程序员写了一个飞机大战
· Obsidian + DeepSeek:免费 AI 助力你的知识管理,让你的笔记飞起来!
· MongoDB 8.0这个新功能碉堡了,比商业数据库还牛
· 白话解读 Dapr 1.15:你的「微服务管家」又秀新绝活了