常用类(四)
JDK1.8新增日期时间类型
如果我们可以跟别人说:“我们在1502643933071见面,别晚了!”那么就再简单不过了。但是我们希望时间与昼夜和四季有关,于是事情就变复杂了。Java1.0中包含了一个Date类,但是它的大多数方法已经在Java 1.1引入Calendar类之后被弃用了。而Calendar并不比Date好多少。它们面临的问题是:
l 可变性:象日期和时间这样的类对象应该是不可变的。Calendar类中可以使用三种方法更改日历字段:set()、add() 和 roll()。
l 偏移性:Date中的年份是从1900开始的,而月份都是从0开始的。
l 格式化:格式化只对Date有用,Calendar则不行。
l 此外,它们也不是线程安全的,不能处理闰秒等。
Date的API: 尽管 Date 类打算反映协调世界时 (UTC),但无法做到如此准确,这取决于 Java 虚拟机的主机环境。当前几乎所有操作系统都假定 1 天 = 24 × 60 × 60 = 86400 秒。但对于 UTC,大约每一两年出现一次额外的一秒,称为“闰秒”。闰秒始终作为当天的最后一秒增加,并且始终在 12 月 31 日或 6 月 30 日增加。例如,1995 年的最后一分钟是 61 秒,因为增加了闰秒。大多数计算机时钟不是特别的准确,因此不能反映闰秒的差别。 |
在类 Date 所有可以接受或返回年、月、日期、小时、分钟和秒值的方法中,将使用下面的表示形式: l 年份 y 由整数 y - 1900 表示。 l 月份由从 0 至 11 的整数表示;0 是一月、1 是二月等等;因此 11 是十二月。 l 日期(一月中的某天)按通常方式由整数 1 至 31 表示。 l 小时由从 0 至 23 的整数表示。因此,从午夜到 1 a.m. 的时间是 0 点,从中午到 1 p.m. 的时间是 12 点。 l 分钟按通常方式由 0 至 59 的整数表示。 l 秒由 0 至 61 的整数表示;值 60 和 61 只对闰秒发生,尽管那样,也只用在实际正确跟踪闰秒的 Java 实现中。于按当前引入闰秒的方式,两个闰秒在同一分钟内发生是极不可能的,但此规范遵循 ISO C 的日期和时间约定。 在所有情形中,针对这些目的赋予方法的参数不需要在指定的范围内;例如,可以把日期指定为 1 月 32 日,并把它解释为 2 月 1 日的相同含义。 |
Date date = new Date(2017-1900,8-1,28); System.out.println(date);//Mon Aug 28 00:00:00 CST 2017 |
可以说,对日期和时间的操作一直是Java程序员最痛苦的地方之一。第三次引入的API是成功的,并且java 8中引入的java.time API 已经纠正了过去的缺陷,将来很长一段时间内它都会为我们服务。
Java 8 吸收了 Joda-Time 的精华,以一个新的开始为 Java 创建优秀的 API。
l java.time – 包含值对象的基础包
l java.time.chrono – 提供对不同的日历系统的访问。
l java.time.format – 格式化和解析时间和日期
l java.time.temporal – 包括底层框架和扩展特性
l java.time.zone – 包含时区支持的类
Java 8 吸收了 Joda-Time 的精华,以一个新的开始为 Java 创建优秀的 API。新的 java.time 中包含了所有关于时钟(Clock),本地日期(LocalDate)、本地时间(LocalTime)、本地日期时间(LocalDateTime)、时区(ZonedDateTime)和持续时间(Duration)的类。历史悠久的 Date 类新增了 toInstant() 方法,用于把 Date 转换成新的表示形式。这些新增的本地化时间日期 API 大大简化了了日期时间和本地化的管理。
说明:大多数开发者只会用到基础包和format包,也可能会用到temporal包。因此,尽管有68个新的公开类型,大多数开发者,大概将只会用到其中的三分之一。
java.time |
注:ISO-8601日历系统是国际标准化组织制定的现代公民的日期和时间的表示法,也就是公历。 |
java.time.chrono |
10.8.1 java.time
1、本地日期(LocalDate)、本地时间(LocalTime)、本地日期时间(LocalDateTime)
LocalDate代表IOS格式(yyyy-MM-dd)的日期,可以存储 生日、纪念日等日期。
LocalTime表示一个时间,而不是日期
LocalDateTime是用来表示日期和时间的,这是一个最常用的类之一。
|
描述 |
now() / now(ZoneId zone) |
静态方法,根据当前时间创建对象/指定时区的对象 |
of() |
静态方法,根据指定日期/时间创建对象 |
getDayOfMonth()/getDayOfYear() |
获得月份天数(1-31) /获得年份天数(1-366) |
getDayOfWeek() |
获得星期几(返回一个 DayOfWeek 枚举值) |
getMonth() |
获得月份, 返回一个 Month 枚举值 |
getMonthValue() / getYear() |
获得月份(1-12) /获得年份 |
getHours()/getMinute()/getSecond() |
获得当前对象对应的小时、分钟、秒 |
withDayOfMonth()/withDayOfYear()/withMonth()/withYear() |
将月份天数、年份天数、月份、年份修改为指定的值并返回新的对象 |
with(TemporalAdjuster t) |
将当前日期时间设置为校对器指定的日期时间 |
plusDays(), plusWeeks(), plusMonths(), plusYears(),plusHours() |
向当前对象添加几天、几周、几个月、几年、几小时 |
minusMonths() / minusWeeks()/minusDays()/minusYears()/minusHours() |
从当前对象减去几月、几周、几天、几年、几小时 |
plus(TemporalAmount t)/minus(TemporalAmount t) |
添加或减少一个 Duration 或 Period |
isBefore()/isAfter() |
比较两个 LocalDate |
isLeapYear() |
判断是否是闰年(在LocalDate类中声明) |
format(DateTimeFormatter t) |
格式化本地日期、时间,返回一个字符串 |
parse(Charsequence text) |
将指定格式的字符串解析为日期、时间 |
//now() @Test public void testLocalDateTime(){ LocalDate date = LocalDate.now(); LocalTime time = LocalTime.now(); LocalDateTime datetime = LocalDateTime.now(); } |
//of()或parse @Test public void testLocalDate() { // LocalDate date = LocalDate.now(); // LocalDate date = LocalDate.of(2017, 3, 20); LocalDate date = LocalDate.parse("2017-03-12"); } |
public static void main(String[] args) { LocalDateTime t = LocalDateTime.now(); System.out.println("这一天是这一年的第几天:"+t.getDayOfYear()); System.out.println("年:"+t.getYear()); System.out.println("月:"+t.getMonth()); System.out.println("月份值:"+t.getMonthValue()); System.out.println("日:"+t.getDayOfMonth()); System.out.println("星期:"+t.getDayOfWeek()); System.out.println("时:"+t.getHour()); System.out.println("分:"+t.getMinute()); System.out.println("秒:"+t.getSecond()); System.out.println(t.getMonthValue()); } |
@Test public void testLocalDate2() { LocalDate date = LocalDate.now();
//withXxx()方法,不改变原来的date对象,返回一个新的对象,不可变性 // LocalDate date2 = date.withDayOfMonth(1);//获取这个月的第一天 LocalDate date2 = date.with(TemporalAdjusters.firstDayOfMonth());// 获取这个月的第一天 System.out.println(date2);
// 获取这个月的最后一天 LocalDate date3 = date.with(TemporalAdjusters.lastDayOfMonth()); System.out.println(date3);
//45天后的日期 LocalDate date4 = date.plusDays(45); System.out.println(date4);
//20天前的日期 LocalDate date5 = date.minusDays(20); System.out.println(date5);
boolean before = date.isBefore(date5); System.out.println(date+"是否比"+date5+"早" + before);
System.out.println(date+"是否是闰年:"+date.isLeapYear());
} |
MonthDay month = MonthDay.of(8, 14); MonthDay today = MonthDay.from(date); System.out.println("今天是否是生日:" + month.equals(today)); |
2、瞬时:Instant
Instant:时间线上的一个瞬时点。 这可能被用来记录应用程序中的事件时间戳。
在处理时间和日期的时候,我们通常会想到年,月,日,时,分,秒。然而,这只是时间的一个模型,是面向人类的。第二种通用模型是面向机器的,或者说是连续的。在此模型中,时间线中的一个点表示为一个很大的数,这有利于计算机处理。在UNIX中,这个数从1970年开始,以秒为的单位;同样的,在Java中,也是从1970年开始,但以毫秒为单位。
java.time包通过值类型Instant提供机器视图。Instant表示时间线上的一点,而不需要任何上下文信息,例如,时区。概念上讲,它只是简单的表示自1970年1月1日0时0分0秒(UTC)开始的秒数。因为java.time包是基于纳秒计算的,所以Instant的精度可以达到纳秒级。
时间戳:指格林威治时间1970年01月01日00时00分00秒(北京时间1970年01月01日08时00分00秒)起至现在的总秒数。
public static void main(String[] args) { Instant t = Instant.now(); System.out.println(t);
//偏移8个小时 OffsetDateTime atOffset = t.atOffset(ZoneOffset.ofHours(8)); System.out.println(atOffset);
long milli = t.toEpochMilli(); System.out.println(milli);
Instant in2 = Instant.ofEpochSecond(10000000); System.out.println(in2); } |
3、带时区的日期、时间的处理
作为一个开发者,如果不用去处理时区和它带来的复杂性,那是幸运的。java.time包下的LocalDate、LocalTime、LocalDateTime和Instant基本能满足需求。当你不可避免时区时,ZonedDateTime等类可以满足我们的需求。
ZonedDateTime:一个在ISO-8601日历系统时区的日期时间,如 2007-12-03T10:15:30+01:00 Europe/Paris。
l 其中每个时区都对应着ID,地区ID都为“{区域}/{城市}”的格式,例如:Asia/Shanghai等
l now():使用系统时间获取当前的ZonedDateTime
l now(ZoneId):返回指定时区的ZonedDateTime
ZoneId:该类中包含了所有的时区信息,一个时区的ID,如 Europe/Paris
l getAvailableZoneIds():静态方法,可以获取所有时区信息
l of(String id):静态方法,用指定的时区信息获取ZoneId对象
Clock:使用时区提供对当前即时、日期和时间的访问的时钟。
public static void main(String[] args) { Set<String> availableZoneIds = ZoneId.getAvailableZoneIds(); for (String string : availableZoneIds) { System.out.println(string); }
ZonedDateTime t = ZonedDateTime.now(); System.out.println(t);
ZonedDateTime t1 = ZonedDateTime.now(ZoneId.of("America/New_York")); System.out.println(t1);
// Clock clock = Clock.systemDefaultZone(); Clock c = Clock.system(ZoneId.of("America/New_York")); System.out.println(c.getZone()); System.out.println(c.instant()); } |
4、持续时间:Duration
Duration:用于计算两个“时间”间隔
public static void main(String[] args) { LocalDateTime t1 = LocalDateTime.now(); LocalDateTime t2 = LocalDateTime.of(2017, 8, 29, 0, 0, 0, 0); Duration between = Duration.between(t1, t2); System.out.println(between);
System.out.println("相差的总天数:"+between.toDays()); System.out.println("相差的总小时数:"+between.toHours()); System.out.println("相差的总分钟数:"+between.toMinutes()); System.out.println("相差的总秒数:"+between.getSeconds()); System.out.println("相差的总毫秒数:"+between.toMillis()); System.out.println("相差的总纳秒数:"+between.toNanos()); System.out.println("不够一秒的纳秒数:"+between.getNano()); } |
5、日期间隔:Period
Period:用于计算两个“日期”间隔
public static void main(String[] args) { LocalDate t1 = LocalDate.now(); LocalDate t2 = LocalDate.of(2018, 12, 31); Period between = Period.between(t1, t2); System.out.println(between);
System.out.println("相差的年数:"+between.getYears());//1年 System.out.println("相差的月数:"+between.getMonths());//又7个月 System.out.println("相差的天数:"+between.getDays());//零25天 System.out.println("相差的总数:"+between.toTotalMonths());//总共19个月 } |
10.8.2 java.time.temporal .TemporalAdjuster : 时间校正器
TemporalAdjuster : 时间校正器。有时我们可能需要获取例如:将日期调整到“下一个工作日”等操作。
TemporalAdjusters : 该类通过静态方法(firstDayOfXxx()/lastDayOfXxx()/nextXxx())提供了大量的常用 TemporalAdjuster 的实现。
public static void main(String[] args) { LocalDate now = LocalDate.now(); System.out.println("下一个周日:"+now.with(TemporalAdjusters.next(DayOfWeek.SUNDAY))); System.out.println("下周五" + now.with(TemporalAdjusters.next(DayOfWeek.FRIDAY))); System.out.println("本月最后一天:"+now.with(TemporalAdjusters.lastDayOfMonth()));//2017-08-31 LocalDate week = LocalDate.now().with(new TemporalAdjuster(){ @Override public Temporal adjustInto(Temporal temporal) { LocalDate date = (LocalDate) temporal; if(date.getDayOfWeek().equals(DayOfWeek.MONDAY)){ return date.plusDays(7); }else if(date.getDayOfWeek().equals(DayOfWeek.TUESDAY)){ return date.plusDays(6); }else if(date.getDayOfWeek().equals(DayOfWeek.WEDNESDAY)){ return date.plusDays(5); }else if(date.getDayOfWeek().equals(DayOfWeek.THURSDAY)){ return date.plusDays(4); }else if(date.getDayOfWeek().equals(DayOfWeek.FRIDAY)){ return date.plusDays(3); }else if(date.getDayOfWeek().equals(DayOfWeek.SATURDAY)){ return date.plusDays(2); }else{ return date.plusDays(1); } } }); System.out.println("下一个工作日:" + week); } |
10.8.3 java.time.format.DateTimeFormatter 类
该类提供了三种格式化方法:
预定义的标准格式。如:ISO_DATE_TIME;ISO_DATE
本地化相关的格式。如:ofLocalizedDate(FormatStyle.MEDIUM)
自定义的格式。如:ofPattern(“yyyy-MM-dd hh:mm:ss”)
public static void main(String[] args) { System.out.println(DateTimeFormatter.ISO_DATE.format(LocalDate.now())); System.out.println(DateTimeFormatter.ISO_DATE_TIME.format(LocalDateTime.now()));
//FULL和SHORT适用于LocalDate和LocalTime System.out.println(DateTimeFormatter.ofLocalizedDate(FormatStyle.FULL).format(LocalDate.now())); System.out.println(DateTimeFormatter.ofLocalizedTime(FormatStyle.SHORT).format(LocalTime.now()));
//LONG和MEDIUM适用于LocalDateTime System.out.println(DateTimeFormatter.ofLocalizedDateTime(FormatStyle.MEDIUM).format(LocalDateTime.now()));
DateTimeFormatter op = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"); System.out.println(op.format(LocalDateTime.now())); } |
10.8.4 与传统日期处理的转换
类 |
To 遗留类 |
From 遗留类 |
java.time.Instant与java.util.Date |
Date.from(instant) |
date.toInstant() |
java.time.Instant与java.sql.Timestamp |
Timestamp.from(instant) |
timestamp.toInstant() |
java.time.ZonedDateTime与java.util.GregorianCalendar |
GregorianCalendar.from(zonedDateTime) |
cal.toZonedDateTime() |
java.time.LocalDate与java.sql.Time |
Date.valueOf(localDate) |
date.toLocalDate() |
java.time.LocalTime与java.sql.Time |
Date.valueOf(localDate) |
date.toLocalTime() |
java.time.LocalDateTime与java.sql.Timestamp |
Timestamp.valueOf(localDateTime) |
timestamp.toLocalDateTime() |
java.time.ZoneId与java.util.TimeZone |
Timezone.getTimeZone(id) |
timeZone.toZoneId() |
java.time.format.DateTimeFormatter与java.text.DateFormat |
formatter.toFormat() |
无 |