在web开发过程中,只要跟数据扯上关系一般都会有日期数据,在java中,也有日期数据类型;在JDK8版本之前,使用Date类型,但使用Date对象中,有很多问题,比如不是现程安全的、直接操作原对象、日期显示错乱(月份从0开始等)等等一系列问题;JDK1.8推出了LocalDateTime等新的日期类型很好的解决了这些问题。
下面内容来源于公众号ImportNew的文章。
Java 8 推出了全新的日期时间API,在教程中我们将通过一些简单的实例来学习如何使用新API。
Java处理日期、日历和时间的方式一直为社区所诟病,将 java.util.Date设定为可变类型,以及SimpleDateFormat的非线程安全使其应用非常受限。
新API基于ISO标准日历系统,java.time包下的所有类都是不可变类型而且线程安全。
Java 8 中的 LocalDate 用于表示当天日期。和java.util.Date不同,它只有日期,不包含时间。当你仅需要表示日期时就用这个类。
package com.shxt.demo02;import java.time.LocalDate;public class Demo01 {public static void main(String[] args) {LocalDate today = LocalDate.now();System.out.println("今天的日期:"+today);}
}
/*运行结果:今天的日期:2018-02-05
*/
package com.shxt.demo02;import java.time.LocalDate;public class Demo02 {public static void main(String[] args) {LocalDate today = LocalDate.now();int year = today.getYear();int month = today.getMonthValue();int day = today.getDayOfMonth();System.out.println("year:"+year);System.out.println("month:"+month);System.out.println("day:"+day);}
}
package com.shxt.demo02;
import java.time.LocalDate;public class Demo03 {public static void main(String[] args) {LocalDate date = LocalDate.of(2018,2,6);System.out.println("自定义日期:"+date);}
}
我们通过静态工厂方法now()非常容易地创建了当天日期,你还可以调用另一个有用的工厂方法LocalDate.of()创建任意日期, 该方法需要传入年、月、日做参数,返回对应的LocalDate实例。这个方法的好处是没再犯老API的设计错误,比如年度起始于1900,月份是从0开 始等等。
package com.shxt.demo02;import java.time.LocalDate;public class Demo04 {public static void main(String[] args) {LocalDate date1 = LocalDate.now();LocalDate date2 = LocalDate.of(2018,2,5);if(date1.equals(date2)){System.out.println("时间相等");}else{System.out.println("时间不等");}}
}
package com.shxt.demo02;import java.time.LocalDate;
import java.time.MonthDay;public class Demo05 {public static void main(String[] args) {LocalDate date1 = LocalDate.now();LocalDate date2 = LocalDate.of(2018,2,6);MonthDay birthday = MonthDay.of(date2.getMonth(),date2.getDayOfMonth());MonthDay currentMonthDay = MonthDay.from(date1);if(currentMonthDay.equals(birthday)){System.out.println("是你的生日");}else{System.out.println("你的生日还没有到");}}
}
只要当天的日期和生日匹配,无论是哪一年都会打印出祝贺信息。你可以把程序整合进系统时钟,看看生日时是否会受到提醒,或者写一个单元测试来检测代码是否运行正确。
package com.shxt.demo02;import java.time.LocalTime;public class Demo06 {public static void main(String[] args) {LocalTime time = LocalTime.now();System.out.println("获取当前的时间,不含有日期:"+time);}
}
可以看到当前时间就只包含时间信息,没有日期。
通过增加小时、分、秒来计算将来的时间很常见。Java 8除了不变类型和线程安全的好处之外,还提供了更好的plusHours()方法替换add(),并且是兼容的。注意,这些方法返回一个全新的LocalTime实例,由于其不可变性,返回后一定要用变量赋值。
package com.shxt.demo02;
import java.time.LocalTime;public class Demo07 {public static void main(String[] args) {LocalTime time = LocalTime.now();LocalTime newTime = time.plusHours(3);System.out.println("三个小时后的时间为:"+newTime);}
}
和上个例子计算3小时以后的时间类似,这个例子会计算一周后的日期。LocalDate日期不包含时间信息,它的plus()方法用来增加天、周、月,ChronoUnit类声明了这些时间单位。由于LocalDate也是不变类型,返回后一定要用变量赋值。
package com.shxt.demo02;import java.time.LocalDate;
import java.time.temporal.ChronoUnit;public class Demo08 {public static void main(String[] args) {LocalDate today = LocalDate.now();System.out.println("今天的日期为:"+today);LocalDate nextWeek = today.plus(1, ChronoUnit.WEEKS);System.out.println("一周后的日期为:"+nextWeek);}
}
可以看到新日期离当天日期是7天,也就是一周。你可以用同样的方法增加1个月、1年、1小时、1分钟甚至一个世纪,更多选项可以查看Java 8 API中的ChronoUnit类。
利用minus()方法计算一年前的日期。
package com.shxt.demo02;
import java.time.LocalDate;
import java.time.temporal.ChronoUnit;public class Demo09 {public static void main(String[] args) {LocalDate today = LocalDate.now();LocalDate previousYear = today.minus(1, ChronoUnit.YEARS);System.out.println("一年前的日期 : " + previousYear);LocalDate nextYear = today.plus(1, ChronoUnit.YEARS);System.out.println("一年后的日期:"+nextYear);}
}
Java 8增加了一个Clock时钟类用于获取当时的时间戳,或当前时区下的日期时间信息。以前用到System.currentTimeInMillis()和TimeZone.getDefault()的地方都可用Clock替换。
package com.shxt.demo02;
import java.time.Clock;public class Demo10 {public static void main(String[] args) {// Returns the current time based on your system clock and set to UTC.Clock clock = Clock.systemUTC();System.out.println("Clock : " + clock.millis());// Returns time based on system clock zoneClock defaultClock = Clock.systemDefaultZone();System.out.println("Clock : " + defaultClock.millis());}
}
另一个工作中常见的操作就是如何判断给定的一个日期是大于某天还是小于某天?在Java 8中,LocalDate类有两类方法isBefore()和isAfter()用于比较日期。
调用isBefore()方法时,如果给定日期小于当前日期则返回true。
package com.shxt.demo02;import java.time.LocalDate;
import java.time.temporal.ChronoUnit;public class Demo11 {public static void main(String[] args) {LocalDate today = LocalDate.now();LocalDate tomorrow = LocalDate.of(2018,2,6);if(tomorrow.isAfter(today)){System.out.println("之后的日期:"+tomorrow);}LocalDate yesterday = today.minus(1, ChronoUnit.DAYS);if(yesterday.isBefore(today)){System.out.println("之前的日期:"+yesterday);}}
}
Java 8不仅分离了日期和时间,也把时区分离出来了。现在有一系列单独的类如ZoneId来处理特定时区,ZoneDateTime类来表示某时区下的时间。这在Java 8以前都是 GregorianCalendar类来做的。下面这个例子展示了如何把本时区的时间转换成另一个时区的时间。
package com.shxt.demo02;import java.time.LocalDateTime;
import java.time.ZoneId;
import java.time.ZonedDateTime;public class Demo12 {public static void main(String[] args) {// Date and time with timezone in Java 8ZoneId america = ZoneId.of("America/New_York");LocalDateTime localtDateAndTime = LocalDateTime.now();ZonedDateTime dateAndTimeInNewYork = ZonedDateTime.of(localtDateAndTime, america );System.out.println("Current date and time in a particular timezone : " + dateAndTimeInNewYork);}
}
与 MonthDay检查重复事件的例子相似,YearMonth是另一个组合类,用于表示信用卡到期日、FD到期日、期货期权到期日等。还可以用这个类得到 当月共有多少天,YearMonth实例的lengthOfMonth()方法可以返回当月的天数,在判断2月有28天还是29天时非常有用。
package com.shxt.demo02;import java.time.*;public class Demo13 {public static void main(String[] args) {YearMonth currentYearMonth = YearMonth.now();System.out.printf("Days in month year %s: %d%n", currentYearMonth, currentYearMonth.lengthOfMonth());YearMonth creditCardExpiry = YearMonth.of(2019, Month.FEBRUARY);System.out.printf("Your credit card expires on %s %n", creditCardExpiry);}
}
package com.shxt.demo02;
import java.time.LocalDate;
public class Demo14 {public static void main(String[] args) {LocalDate today = LocalDate.now();if(today.isLeapYear()){System.out.println("This year is Leap year");}else {System.out.println("2018 is not a Leap year");}}
}
有一个常见日期操作是计算两个日期之间的天数、周数或月数。在Java 8中可以用java.time.Period类来做计算。下面这个例子中,我们计算了当天和将来某一天之间的月数。
package com.shxt.demo02;
import java.time.LocalDate;
import java.time.Period;public class Demo15 {public static void main(String[] args) {LocalDate today = LocalDate.now();LocalDate java8Release = LocalDate.of(2018, 12, 14);Period periodToNextJavaRelease = Period.between(today, java8Release);System.out.println("Months left between today and Java 8 release : "+ periodToNextJavaRelease.getMonths() );}
}
Instant类有一个静态工厂方法now()会返回当前的时间戳,如下所示:
package com.shxt.demo02;
import java.time.Instant;public class Demo16 {public static void main(String[] args) {Instant timestamp = Instant.now();System.out.println("What is value of this instant " + timestamp.toEpochMilli());}
}
时间戳信息里同时包含了日期和时间,这和java.util.Date很像。实际上Instant类确实等同于 Java 8之前的Date类,你可以使用Date类和Instant类各自的转换方法互相转换。
例如:Date.from(Instant) 将Instant转换成java.util.Date,Date.toInstant()则是将Date类转换成Instant类。
package com.shxt.demo02;
import java.time.LocalDate;
import java.time.format.DateTimeFormatter;public class Demo17 {
public static void main(String[] args) {String dayAfterTommorrow = "20180205";LocalDate formatted = LocalDate.parse(dayAfterTommorrow,DateTimeFormatter.BASIC_ISO_DATE);System.out.println(dayAfterTommorrow+" 格式化后的日期为: "+formatted);}}
package com.shxt.demo02;
import java.time.LocalDate;
import java.time.LocalDateTime;
import java.time.format.DateTimeFormatter;public class Demo18 {public static void main(String[] args) {LocalDateTime date = LocalDateTime.now();DateTimeFormatter format1 = DateTimeFormatter.ofPattern("yyyy/MM/dd HH:mm:ss");//日期转字符串String str = date.format(format1);System.out.println("日期转换为字符串:" + str);DateTimeFormatter format2 = DateTimeFormatter.ofPattern("yyyy/MM/dd HH:mm:ss");//字符串转日期LocalDate date2 = LocalDate.parse(str, format2);System.out.println("日期类型:" + date2);}
}