文章目录
新的日期API
一 问题发现
Java
原本自带的java.util.Date和java.util.Calendar类
,实际上两种类有线程不安全的风险(虽然学习的时候处于单线程环境并不会出现问题),但是之后到了企业中还是可能会增加学习成本,重新学习如何处理时间,所以推出了这个Java8
的最新时间类库的讲解,希望降低学生的学习成本,能够更快的融入到企业开发实战中去。
1.1 问题
- 时间转换过程繁琐,代码易冗余
- 在多线程情况下,会出现线程不安全问题
- 老版本API使用不规范
多线程
package 日期.老版本;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
/**
* @author shu
* @date 2021/6/3
* @description 多线程不加锁情况下
*/
public class SingleDateText {
//日期格式化格式
static SimpleDateFormat simpleDateFormat=new SimpleDateFormat("yyyy-MM-dd");
public static void main(String[] args) {
//模拟10个线程
for (int i = 0; i < 10; i++) {
new Thread(new Runnable() {
@Override
public void run() {
try {
// synchronized (simpleDateFormat) {
Date date = simpleDateFormat.parse("2021-01-12");
System.out.println(date);
//}
} catch (ParseException e) {
e.printStackTrace();
}
}
}).start();
}
}
}
1.2 解决方法
Java8
中提供的java.time
包中的常用日期与相关方法
-
Instant类
:对时间轴上的单一瞬时点建模,可以用于记录应用程序中的事件时间戳,之后学习的类型转换中,均可以使用Instant类作为中间类完成转换。
-
Duration
类表示秒或纳秒时间间隔,适合处理较短的时间,需要更高的精确性。
-
Period
类表示一段时间的年、月、日。
-
LocalDate
类是一个不可变的日期时间对象,表示日期,通常被视为年月日。
-
LocalTime
类是一个不可变的日期时间对象,代表一个时间,通常被看作是小时-秒,时间表示为纳秒精度。
-
LocalDateTime
类是一个不可变的日期时间对象,代表日期时间,通常被视为年-月-日=时-分-秒。
-
ZonedDateTime
类是具有时区的日期时间的不可变表示,此类存储所有日期和时间字段,精度为纳秒,时区为区域偏移量,用于处理模糊的本地日期时间。
-
now
方法在日期/时间类的使用 -
Year
类(表示年) -
YearMonth
类(表示年月) -
MonthDay
类(表示月日) -
Date-Time API中的所有类均生成不可变实例,它们是线程安全的,并且这些类不提供公共构造函数,也就是说没办法通过new的方式直接创建,需要采用工厂方法加以实例化。
二 基本使用
2.1 now方法(系统时间)
package 日期.新版本;
import java.time.*;
/**
* @author shu
* @date 2021/6/4
* @description now方法创建实例
*/
public class NowText {
public static void main(String[] args) {
//国际标准时间
Instant instant = Instant.now();
//年月日
LocalDate localDate = LocalDate.now();
//年月日时分秒
LocalDateTime localDateTime = LocalDateTime.now();
//时间+时区
ZonedDateTime zonedDateTime = ZonedDateTime.now();
//年
Year year = Year.now();
//年月
YearMonth yearMonth = YearMonth.now();
//月日
MonthDay day = MonthDay.now();
System.out.println("instant==>"+instant);
System.out.println("localDate==>"+localDate);
System.out.println("localDateTime==>"+localDateTime);
System.out.println("zonedDateTime==>"+zonedDateTime);
System.out.println("year==>"+year);
System.out.println("yearMonth==>"+yearMonth);
System.out.println("day==>"+day);
}
}
2.2 of方法(指定时间)
of
方法可以根据给定的参数生成对应的日期/时间对象,基本上每个基本类都有of
方法用于生成的对应的对象,而且重载形式对边,可以根据不同的参数生成对应的数据。
package 日期.新版本;
import java.time.LocalDate;
import java.time.LocalDateTime;
import java.time.Month;
/**
* @author shu
* @date 2021/6/4
* @description Of方法
*/
public class OfText {
public static void main(String[] args) {
//普通方法
LocalDate localDate1 = LocalDate.of(2012, 5, 12);
//枚举方法
LocalDate localDate2 = LocalDate.of(2012, Month.AUGUST, 12);
//指定方法
LocalDateTime localDateTime = LocalDateTime.of(2021, 12, 5, 12, 5);
System.out.println(localDate1);
System.out.println(localDate2);
System.out.println(localDateTime);
}
}
2.3 ZonedId时区(国际化)
package 日期.新版本;
import java.time.LocalDateTime;
import java.time.ZoneId;
import java.time.ZonedDateTime;
/**
* @author shu
* @date 2021/6/4
* @description 时区
*/
public class ZonedIdText {
public static void main(String[] args) {
//本地时间
LocalDateTime localDateTime = LocalDateTime.of(2021, 12, 5, 12, 5);
//东京时区
ZonedDateTime zonedDateTime = localDateTime.atZone(ZoneId.of("Asia/Tokyo"));
System.out.println(zonedDateTime);
}
}
2.4 plus方法 (增加) minus方法(减少)
-
LocalDate plusDay(long days)
增加天数 -
LocalDate plusWeeks(long weeks)
增加周数 -
LocallDate plusMonths(long months)
增加月数 -
LocalDate plusYears(long years)
增加年数 -
LocalTime plusNanos(long nanos)
增加纳秒 -
LocalTime plusSeconds(long seconds)
增加秒 -
LocalTime plusMinutes(long minutes)
增加分钟 -
LocalTime plusHours(long hours)
增加小时
package 日期.新版本;
import java.time.LocalDate;
import java.time.LocalDateTime;
/**
* @author shu
* @date 2021/6/4
* @description 增加(减少一样的)
*/
public class PlusText {
public static void main(String[] args) {
LocalDate localDate = LocalDate.now();
//年
LocalDate plusYears = localDate.plusYears(1);
//月
LocalDate plusMonths = localDate.plusMonths(2);
//周
LocalDate plusWeeks = localDate.plusWeeks(1);
//日
LocalDate plusDays = localDate.plusDays(1);
LocalDateTime localDateTime = LocalDateTime.now();
//时
LocalDateTime plusHours = localDateTime.plusHours(1);
//分
LocalDateTime plusMinutes = localDateTime.plusMinutes(1);
//秒
LocalDateTime plusSeconds = localDateTime.plusSeconds(1);
System.out.println(plusYears);
System.out.println(plusMonths);
System.out.println(plusWeeks);
System.out.println(plusDays);
System.out.println(plusHours);
System.out.println(plusMinutes);
System.out.println(plusSeconds);
}
}
2.5 With 方法修改日期
-
LocalDateTime withNano(int i)
修改纳秒 -
LocalDateTime withSecond(int i)
修改秒 -
LocalDateTime withMinute(int i)
修改分支 -
LocalDateTime withHour(int i)
修改小时 -
LocalDateTime withDayOfMonth(int i)
修改日 -
LocalDateTime withMonth(int i)
修改月 -
LocalDateTime withYear(int i
) 修改年
package 日期.新版本;
import java.time.LocalDate;
import java.time.temporal.ChronoField;
/**
* @author shu
* @date 2021/6/4
* @description 修改日期
*/
public class WithText {
public static void main(String[] args) {
LocalDate localDate = LocalDate.now();
LocalDate withYear = localDate.withYear(2020);
LocalDate withMonth = localDate.withMonth(1);
LocalDate withDayOfYear = localDate.withDayOfYear(20);
LocalDate withDayOfMonth = localDate.withDayOfMonth(2);
LocalDate date = localDate.with(ChronoField.ALIGNED_DAY_OF_WEEK_IN_MONTH, 1);
System.out.println(withYear);
System.out.println(withMonth);
System.out.println(withDayOfYear);
System.out.println(withDayOfMonth);
System.out.println(date);
}
}
三 调节器
3.1 TemporalAdjusters类中常用静态方法
-
static TemporalAdjuster firstDayofNextMonth()
下个月的第一天 -
static TemporalAdjuster firstDayOfNextYear()
下一年的第一天 -
static TemporalAdjuster firstDayOfYear()
当年的第一天
package 日期.新版本;
import java.time.LocalDate;
import java.time.temporal.TemporalAdjusters;
/**
* @author shu
* @date 2021/6/4
* @description 调度器的使用
*/
public class TemporalAdjustersText {
public static void main(String[] args) {
//一年中的第一天
LocalDate localDate1 = LocalDate.now().with(TemporalAdjusters.firstDayOfYear());
//一月的第一天
LocalDate localDate2 = LocalDate.now().with(TemporalAdjusters.firstDayOfMonth());
//下个年的第一天
LocalDate localDate3 = LocalDate.now().with(TemporalAdjusters.firstDayOfNextYear());
//一月的最后一天
LocalDate localDate4 = LocalDate.now().with(TemporalAdjusters.lastDayOfMonth());
//一年的最后一天
LocalDate localDate5 = LocalDate.now().with(TemporalAdjusters.lastDayOfYear());
System.out.println(localDate1);
System.out.println(localDate2);
System.out.println(localDate3);
System.out.println(localDate4);
System.out.println(localDate5);
}
}
3.2 DayOfWeek枚举类使用
package 日期.新版本;
import java.time.DayOfWeek;
import java.time.LocalDate;
import java.time.temporal.TemporalAdjusters;
/**
* @author shu
* @date 2021/6/4
* @description DayOfWeek枚举类使用
*/
public class TemporalAdjustersText {
public static void main(String[] args) {
//下一周星期几
LocalDate localDate6 = LocalDate.now().with(TemporalAdjusters.next(DayOfWeek.SUNDAY));
//上一周星期几
LocalDate localDate7 = LocalDate.now().with(TemporalAdjusters.previous(DayOfWeek.SUNDAY));
System.out.println(localDate6);
System.out.println(localDate7);
}
}
3.3 自定义调节器的应用
package 日期.新版本;
import java.time.DayOfWeek;
import java.time.LocalDate;
import java.time.temporal.Temporal;
import java.time.temporal.TemporalAdjuster;
import java.time.temporal.TemporalAdjusters;
/**
* @author shu
* @date 2021/6/4
* @description 自定义调节器
* 目标:如果每个月的15号是星期六或者星期天,将发工资日期调整到上周的星期五
*/
public class PayTemporalAdjuster implements TemporalAdjuster {
@Override
public Temporal adjustInto(Temporal temporal) {
//解析日期
LocalDate localDate = LocalDate.from(temporal);
int days;
//计算是否是15号
if(localDate.getDayOfMonth()!=15){
days=15;
}else{
days=localDate.getDayOfMonth();
}
//修改日期
LocalDate RealDay = localDate.withDayOfMonth(days);
//判断是否是星期六或星期天
if(RealDay.getDayOfWeek()== DayOfWeek.SUNDAY||RealDay.getDayOfWeek()==DayOfWeek.SATURDAY){
RealDay.with(TemporalAdjusters.previous(DayOfWeek.FRIDAY));
}
//返回日期
return RealDay;
}
}
3.4 TemporalQuery的应用
package 日期.新版本;
import java.time.LocalDate;
import java.time.Month;
import java.time.temporal.ChronoUnit;
import java.time.temporal.TemporalAccessor;
import java.time.temporal.TemporalQuery;
/**
* @author shu
* @date 2021/6/4
* @description 计算某一个日期距离5月1日还有多少天
*/
public class DaysTemporalQuery implements TemporalQuery<Long> {
@Override
public Long queryFrom(TemporalAccessor temporal) {
//格式化时间
LocalDate localDate = LocalDate.from(temporal);
//封装5月1号
LocalDate date = LocalDate.of(localDate.getYear(), Month.MAY, 1);
//判断是否超过5月1号
if(localDate.isAfter(date)){
date = date.plusYears(1);
}
//计算时间差返回
return ChronoUnit.DAYS.between(localDate, date);
}
}
四 新旧转换
4.1 Instant时间转换
-
使用Instant类将java.util.Date转换为java.time.LocalDate
-
java.sql.Date类中提供直接转换为LocalDate的方法,toLocalDate()
package 日期.日期转换;
import java.sql.Timestamp;
import java.time.*;
import java.util.Date;
/**
* @author shu
* @date 2021/6/4
* @description Instant时间转换
*/
public class ChangeLocalDate {
public static void main(String[] args) {
// java.util.Date;
Date date = new Date();
Instant instant = date.toInstant();
ZonedDateTime zonedDateTime = instant.atZone(ZoneId.systemDefault());
LocalDate localDate = zonedDateTime.toLocalDate();
System.out.println("之前的时间==》"+date);
System.out.println("转换的时间==》"+localDate);
//java.sql.Timestamp;
java.sql.Date date1 = new java.sql.Date(System.currentTimeMillis());
LocalDate localDate1 = date1.toLocalDate();
System.out.println("之前的时间==》"+date1);
System.out.println("转换的时间==》"+localDate1);
//java.sql.Timestamp;
Timestamp timestamp = new Timestamp(System.currentTimeMillis());
LocalDateTime dateTime = timestamp.toLocalDateTime();
System.out.println("之前的时间==》"+timestamp);
System.out.println("转换的时间==》"+dateTime);
}
}
4.2 时间戳转换
package 日期.日期转换;
import java.time.LocalDate;
import java.util.Date;
/**
* @author shu
* @date 2021/6/4
* @description 时间戳转换
*/
public class TimeLocalDate {
public static void main(String[] args) {
long time = new Date().getTime();
java.sql.Date date = new java.sql.Date(time);
LocalDate localDate = date.toLocalDate();
System.out.println("之前的时间==》"+date);
System.out.println("转换的时间==》"+localDate);
}
}
4.3 日历转换
package 日期.日期转换;
import java.time.LocalDate;
import java.time.ZoneId;
import java.time.ZonedDateTime;
import java.util.Calendar;
import java.util.TimeZone;
/**
* @author shu
* @date 2021/6/4
* @description 日历转换
*/
public class ChangeCalendar {
public static void main(String[] args) {
Calendar calendar = Calendar.getInstance();
//获取时区
TimeZone zone = calendar.getTimeZone();
//获取时区id
ZoneId id = zone.toZoneId();
//转换
ZonedDateTime zonedDateTime = ZonedDateTime.ofInstant(calendar.toInstant(), id);
LocalDate localDate = zonedDateTime.toLocalDate();
System.out.println("之前的时间==》"+calendar);
System.out.println("转换的时间==》"+localDate);
}
}
4.4 日期格式化
SimpleDateFormat
类在刚开始的时候讲过了是线程不安全的,所以Java8
提供了新的格式化类 DateTimeFormatter
package 日期.日期转换;
import java.time.LocalDateTime;
import java.time.format.DateTimeFormatter;
/**
* @author shu
* @date 2021/6/4
* @description 日期格式化
*/
public class ToStringLocalDate {
public static void main(String[] args) {
LocalDateTime localDate = LocalDateTime.now();
// localDate.format(DateTimeFormatter.ofPattern("yyyy-MM-dd"));
// yyyy-MM-dd
String s1 = localDate.format(DateTimeFormatter.ISO_DATE);
String s2 = localDate.format(DateTimeFormatter.ISO_DATE_TIME);
System.out.println(s1);
System.out.println(s2);
}
}