1、JDK8之前日期时间API
1.1、java.lang.System类
System类提供的 public static long currentTimeMillis()
用来返回当前时间与1970年1月1日0时0分0秒之间以毫秒为单位的时间差。 此方法适于计算时间差,即时间戳,long型 。
计算世界时间的主要标准有:
- UTC(Coordinated Universal Time)
- GMT(Greenwich Mean Time)
- CST(Central Standard Time)
public void test1() {
StringBuffer buffer = new StringBuffer("");
//1. 返回当前时间与1970年1月1日0时0分0秒之间以毫秒为单位的时间差。
//称为时间戳
long startTime = System.currentTimeMillis();//1645496062871
for(int i = 0;i<20000;i++){
buffer.append(String.valueOf(i));}
long endTime = System.currentTimeMillis();
System.out.println("StringBuffer的执行时间:"+(endTime-startTime));//8
}
1.2、java.util.Date类
java.util.Date
类是 java.sql.Date
类的父类,表示从 1970 年 1 月 1 日起的毫秒数
构造器:
Date()
:创建一个对应当前时间的Date对象**。 **Date(long date)
:创建指定毫秒数的Date对象
常用方法
getTime()
:返回自 1970 年 1 月 1 日 00:00:00 GMT 以来此 Date 对象对应的时间戳,即毫秒数toString()
:显示当前的年、月、日、时、分、秒。
public void test2(){
//构造器一:Date():创建一个对应当前时间的Date对象
Date date1 = new Date(); //util下的date
System.out.println(date1.toString());//Sat Feb 16 16:35:31 GMT+08:00 2019
System.out.println(date1.getTime());//1550306204104
//构造器二:创建指定毫秒数的Date对象
Date date2 = new Date(155030620410L);//util下的date
System.out.println(date2.toString());
}
备注:对于public Date(int year,int month,int date)
这个构造器已经过时(@Deprecated
),但是也能用
其他方法
Date 对象创建以后,可以调用下面的方法。
序号 | 方法和描述 |
---|---|
1 | boolean after(Date date) 若当调用此方法的Date对象在指定日期之后返回true,否则返回false。 |
2 | boolean before(Date date) 若当调用此方法的Date对象在指定日期之前返回true,否则返回false。 |
3 | Object clone( ) 返回此对象的副本。 |
4 | int compareTo(Date date) 比较当调用此方法的Date对象和指定日期。两者相等时候返回0。调用对象在指定日期之前则返回负数。调用对象在指定日期之后则返回正数。 |
5 | int compareTo(Object obj) 若obj是Date类型则操作等同于compareTo(Date) 。否则它抛出ClassCastException。 |
6 | boolean equals(Object date) 当调用此方法的Date对象和指定日期相等时候返回true,否则返回false。 |
7 | long getTime( ) 返回自 1970 年 1 月 1 日 00:00:00 GMT 以来此 Date 对象表示的毫秒数。 |
8 | int hashCode( ) 返回此对象的哈希码值。 |
9 | void setTime(long time) 用自1970年1月1日00:00:00 GMT以后time毫秒数设置时间和日期。 |
10 | String toString( ) 把此 Date 对象转换为以下形式的 String: dow mon dd hh:mm:ss zzz yyyy 其中: dow 是一周中的某一天 (Sun, Mon, Tue, Wed, Thu, Fri, Sat)。 |
日期比较
Java使用以下三种方法来比较两个日期:
- 使用 getTime() 方法获取两个日期(自1970年1月1日经历的毫秒数值),然后比较这两个值。
- 使用方法 before(),after() 和 equals()。例如,一个月的12号比18号早,则 new Date(99, 2, 12).before(new Date (99, 2, 18)) 返回true。
- 使用 compareTo() 方法,它是由 Comparable 接口定义的,Date 类实现了这个接口。
1.3、java.sql.Date类
java.sql.Date
对应着数据库中的日期类型的变量
如何实例化
1. //创建java.sql.Date对象
2. java.sql.Date date3 = new java.sql.Date(35235325345L);
3. System.out.println(date3);//1971-02-13 自己的tostring方法
如何将java.util.Date对象转换为java.sql.Date对象
//情况一:
Date date4 = new java.sql.Date(2343243242323L); //多态
java.sql.Date date5 = (java.sql.Date) date4; //强转
//情况二:
Date date6 = new Date();
java.sql.Date date7 = new java.sql.Date(date6.getTime());
如何将java.sql.Date对象转换为java.util.Date对象
由于是子父类关系,所以直接使用多态即可
1.4、SimpleDateFormat类【不建议使用】
日期格式化建议使用 DateTimeFormatter 类
Date
类的API不易于国际化,大部分被废弃了,java.text.SimpleDateFormat 类是一个不与语言环境有关的方式来格式化和解析日期的具体类。SimpleDateFormat
允许你选择任何用户自定义日期时间格式来运行
它允许进行格式化:日期—>文本、解析:文本—>日期
格式化:
SimpleDateFormat()
:默认的模式和语言环境创建对象public SimpleDateFormat(String pattern)
:该构造方法可以用参数pattern
指定的格式创建一个格式化对象- 上面两种构造器方式创建的
SimpleDateFormat
对象调用public String format(Date date)
方法可以格式化时间对象date
解析:
public Date parse(String source)
:从给定字符串的开始解析文本,以生成一个日期
SimpleDateFormat的实例化
格式化解析1
1. //实例化SimpleDateFormat:使用默认的构造器
2. SimpleDateFormat sdf = new SimpleDateFormat();
3.
4. //格式化:日期 --->字符串
5. Date date = new Date();
6. System.out.println(date);
7.
8. String format = sdf.format(date);
9. System.out.println(format);
10.
11. //解析:格式化的逆过程,字符串 ---> 日期
12. String str = "19-12-18 上午11:43";
13. Date date1 = sdf.parse(str); //防止输入的日期字符串不合法,要throws抛异常
14. System.out.println(date1);
记住:这就是默认的行为,使用怎样的格式去格式化日期,就必须使用怎样的格式去解析,一般不使用这种默认的构造器
格式化解析2
//按照指定的方式格式化和解析:调用带参的构造器
//SimpleDateFormat sdf1 = new SimpleDateFormat("yyyyy.MMMMM.dd GGG hh:mm aaa");
SimpleDateFormat sdf1 = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss");
//格式化
String format1 = sdf1.format(date);
System.out.println(format1);//2019-02-18 11:48:27
//解析:要求字符串必须是符合SimpleDateFormat识别的格式(通过构造器参数体现),
//否则,抛异常
Date date2 = sdf1.parse("2020-02-18 11:48:27");
System.out.println(date2);
这一行代码确立了转换的格式,其中 yyyy 是完整的公元年,MM 是月份,dd 是日期,HH:mm:ss 是时、分、秒。
注意: 有的格式大写,有的格式小写,例如 MM 是月份,mm 是分;HH 是 24 小时制,而 hh 是 12 小时制。
日期和时间的格式化编码
时间模式字符串用来指定时间格式。在此模式中,所有的 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 | 时区 | Eastern Standard Time |
’ | 文字定界符 | Delimiter |
" | 单引号 | ` |
练习
1:字符串"2020-09-08"转换为 java.sql.Date
@Test
public void testExer() throws ParseException {
String birth = "2020-09-08";
SimpleDateFormat sdf1 = new SimpleDateFormat("yyyy-MM-dd");
Date date = sdf1.parse(birth); //解析成util下的date
//System.out.println(date);
java.sql.Date birthDate = new java.sql.Date(date.getTime());
System.out.println(birthDate);
}
1.5、java.util.Calendar(日历)类
我们现在已经能够格式化并创建一个日期对象了,但是我们如何才能设置和获取日期数据的特定部分呢,比如说小时,日,或者分钟? 我们又如何在日期的这些部分加上或者减去值呢? 答案是使用Calendar 类。Calendar类的功能要比Date类强大很多,而且在实现方式上也比Date类要复杂一些。Calendar类是一个抽象类,所以不能实例化,只能用它的子类去实例化,创建对象的过程对程序员来说是透明的,只需要使用getInstance
方法创建即可。
获取Calendar实例的方法
- 使用
Calendar.getInstance()
方法 - 调用它的子类
GregorianCalendar
的构造器。
实例化
public void testCalendar(){
//方式一:创建其子类(GregorianCalendar)的对象
//方式二:调用其静态方法getInstance()
Calendar calendar = Calendar.getInstance();//默认是当前日期
//System.out.println(calendar.getClass()); //java.util.GregorianCalendar
}
常用方法
//get()
int days = calendar.get(Calendar.DAY_OF_MONTH); //获取这个月的第几天
System.out.println(days);
System.out.println(calendar.get(Calendar.DAY_OF_YEAR));
//set()
//calendar可变性
//calendar.set(2009, 6, 12);//把Calendar对象的年月日分别设这为:2009、6、12
calendar.set(Calendar.DAY_OF_MONTH,22); //可以随意设置这个月的第几天
days = calendar.get(Calendar.DAY_OF_MONTH);
System.out.println(days);
//add()
calendar.add(Calendar.DAY_OF_MONTH,-3);
days = calendar.get(Calendar.DAY_OF_MONTH);
System.out.println(days);
//getTime():日历类---> Date
Date date = calendar.getTime();
System.out.println(date);
//setTime():Date ---> 日历类
Date date1 = new Date(); //获取当前时间
calendar.setTime(date1);
days = calendar.get(Calendar.DAY_OF_MONTH);
System.out.println(days);
一个Calendar
的实例是系统时间的抽象表示,通过get(int field)
方法来取得想 要的时间信息。比如YEAR、MONTH、DAY_OF_WEEK、HOUR_OF_DAY 、 MINUTE、SECOND
注意:
- 获取月份时:一月是0,二月是1,以此类推,12月是11
- 获取星期时:周日是1,周二是2 , 。。。。周六是7
Calendar类对象字段类型
Calendar类中用以下这些常量表示不同的意义,jdk内的很多类其实都是采用的这种思想
常量 | 描述 |
---|---|
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 | 星期几 |
2、JDK8中新日期时间API
新日期时间API出现的背景
如果我们可以跟别人说:“我们在1502643933071见面,别晚了!”那么就再简单不 过了。但是我们希望时间与昼夜和四季有关,于是事情就变复杂了。JDK 1.0中包含了 一个java.util.Date
类,但是它的大多数方法已经在JDK 1.1引入Calendar
类之后被弃用 了。而Calendar并不比Date好多少。它们面临的问题是:
- 可变性:像日期和时间这样的类应该是不可变的。
- 偏移性:Date中的年份是从1900开始的,而月份都从0开始。
- 格式化:格式化只对Date有用,Calendar则不行。
- 此外,它们也不是线程安全的;不能处理闰秒等。
@Test
public void testDate(){
//偏移量,指定年月日的日期构造器,2020.9.8
Date date1 = new Date(2020 - 1900,9 - 1,8); 老版本的date有偏移量。必须加上才能获取到正确的日期
System.out.println(date1);//Tue Sep 08 00:00:00 GMT+08:00 2020
}
Java 8 吸收了 Joda-Time
的精华,以一个新的开始为 Java 创建优秀的 API。 新的 java.time
中包含了所有关于本地日期(LocalDate)、本地时间 (LocalTime)、本地日期时间(LocalDateTime)、时区(ZonedDateTime) 和持续时间(Duration)的类。历史悠久的 Date 类新增了 toInstant()
方法, 用于把 Date 转换成新的表示形式。这些新增的本地化时间日期 API 大大简化了日期时间和本地化的管理。
2.1、常用类
LocalDate、LocalTime、LocalDateTime 类是其中较重要的几个类,它们的实例是不可变的对象,分别表示使用 ISO-8601日历系统的日期、时间、日期和时间。 它们提供了简单的本地日期或时间,并不包含当前的时间信息,也不包含与时区相关的信息。
LocalDate
代表IOS格式(yyyy-MM-dd)的日期,可以存储生日、纪念日等日期。LocalTime
表示一个时间,而不是日期。LocalDateTime
是用来表示日期和时间的,这是一个最常用的类之一。类似于Calendar
2.2、常用方法
now():获取当前的日期、时间、日期+时间
@Test
public void test1(){
//now():获取当前的日期、时间、日期+时间
LocalDate localDate = LocalDate.now();
LocalTime localTime = LocalTime.now();
LocalDateTime localDateTime = LocalDateTime.now();
System.out.println(localDate);
System.out.println(localTime);
System.out.println(localDateTime);
}
of():设置指定的年、月、日、时、分、秒,没有偏移量
LocalDateTime localDateTime1 = LocalDateTime.of(2020, 10, 6, 13, 23, 43);
System.out.println(localDateTime1);
getXxx():获取相关的属性
System.out.println(localDateTime.getDayOfMonth());
System.out.println(localDateTime.getDayOfWeek());
System.out.println(localDateTime.getMonth());
System.out.println(localDateTime.getMonthValue());
System.out.println(localDateTime.getMinute());
withXxx():设置相关的属性
//体现不可变性
//withXxx():设置相关的属性
LocalDate localDate1 = localDate.withDayOfMonth(22);
System.out.println(localDate);
System.out.println(localDate1);
不可变性
//不可变性
LocalDateTime localDateTime3 = localDateTime.plusMonths(3);
System.out.println(localDateTime);
System.out.println(localDateTime3);
LocalDateTime localDateTime4 = localDateTime.minusDays(6);
System.out.println(localDateTime);
System.out.println(localDateTime4);
3、instant 瞬时类
java.time.Instant
:时间线上的一个瞬时点。 这可能被用来记录应用程序中的事件时间戳【时间戳是指格林威治时间1970年01月01日00时00分00秒(北京时间1970年01月01 日08时00分00秒)起至现在的总秒数】。
Instant
的使用类似于 java.util.Date
类
相关方法
@Test
public void test2(){
//now():获取本初子午线对应的标准时间
Instant instant = Instant.now();
System.out.println(instant);//2019-02-18T07:29:41.719Z
//添加时间的偏移量
OffsetDateTime offsetDateTime = instant.atOffset(ZoneOffset.ofHours(8));
System.out.println(offsetDateTime);//2019-02-18T15:32:50.611+08:00
//toEpochMilli():获取自1970年1月1日0时0分0秒(UTC)开始的毫秒数 ---> Date类的getTime()
long milli = instant.toEpochMilli();
System.out.println(milli);
//ofEpochMilli():通过给定的毫秒数,获取Instant实例 -->Date(long millis)
Instant instant1 = Instant.ofEpochMilli(1550475314878L);
System.out.println(instant1);
}
4、DateTimeFormatter类【建议使用】
4.1、格式化与解析日期或时间
java.time.format.DateTimeFormatter
类:该类提供了三种格式化方法:
- 预定义的标准格式,如:
ISO_LOCAL_DATE_TIME
;ISO_LOCAL_DATE
;ISO_LOCAL_TIME
- 本地化相关的格式。如:
ofLocalizedDateTime(FormatStyle.LONG)
- 自定义的格式。如:
ofPattern(“yyyy-MM-dd hh:mm:ss”)
所以DateTimeFormatter
类就是用于格式化或解析日期、时间,类似于SimpleDateFormat
4.2、常用API
格式解析1
//方式一预定义的标准格式。
DateTineFornatter formatter = DateTineFornatter.ISO_LOCAL_DATE_TIME;
//格式化:日期-->字符申
LocalDateTime LocalDateTime = LocalDateTime.now();//日期
String str1 = formatter.format(LocalDateTime);
System.out.printLn(locaLDateTine);
System.out.println(str1);//2819-82-18T15:42:18.797
//解析:字持串-->日期
TenporalAccessor parse = formatter.parse("2019-02-18T15:42:18.797");
System.out.printLn(parse);
格式解析2
//方式二:
//本地化相关的格式。如: ofLocalizedDateTime()
//FormatStyle.LONG / FormatStyle.MEDIUN / FormatStyle.SHORT : 适用LocalDateTime
LocalDateTime LocalDateTime = LocalDateTime.now();
DateTimeFormatter formatter1 = DateTimeFormatter.ofLocalizedDateTime(Formatstyle.LONG);
//格式化
String str2 = formatter1.format(localDateTime);
System.out.printLn(str2);//2019年2月18日下午03时47分16秒
//本地化相关的格式。如: ofLocalizedDate()
//Formatstyle.FULL / Formatstyle.LONG / Formatstyle .MEDIUM / Formatstyle.SHORT :适用fLocalDate
DateTimeFormatter formatter2 = DateTimeFormatter.ofLocalizedDate(FormatStyle.MEDIUM);
//格式化
String str3 = formatter2.format(LocalDate.now());
System.out.println(str3);//2819-2-18
格式解析3
//重点:方式三: 自定义的格式。如: ofPattern( "yyyy-MM-dd hh:mm:ss")
DateTimeFormatter formatter3 = DateTimeFormatter.ofPattern("yyyy-M-dd hh:mm:ss");
//格式化
String str4 = formatter3.format(LocalDateTime.now());
System.out.println(str4);//2019-02-18 03:52:09
//解析
TemporalAccessor accessor = formatter3.parse("2019-02-18 03:52:09");
System.out.println(accessor);
Style.MEDIUM);
//格式化
String str3 = formatter2.format(LocalDate.now());
System.out.println(str3);//2819-2-18
格式解析3
//重点:方式三: 自定义的格式。如: ofPattern( "yyyy-MM-dd hh:mm:ss")
DateTimeFormatter formatter3 = DateTimeFormatter.ofPattern("yyyy-M-dd hh:mm:ss");
//格式化
String str4 = formatter3.format(LocalDateTime.now());
System.out.println(str4);//2019-02-18 03:52:09
//解析
TemporalAccessor accessor = formatter3.parse("2019-02-18 03:52:09");
System.out.println(accessor);