JDK8--09:全新的时间API
在JDK8之前,时间有各种问题,最大的问题就是,我们使用的时间格式化类SimpleDateFormat不是线程安全的
为了更准确的说明SimpleDateFormat非线程安全,演示一个并发做时间格式化的操作
public void test() throws Exception{
//全新的时间API 都不是线程安全的
SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyyMMdd");
Callable<Date> call = new Callable<Date>() {
@Override
public Date call() throws Exception {
return simpleDateFormat.parse("");
}
};
ExecutorService newFixedThreadPool = Executors.newFixedThreadPool();
List<Future<Date>> list = new ArrayList<>();
for(int i=;i<;i++){
list.add(newFixedThreadPool.submit(call));
}
for (Future<Date> future : list){
log.info("============={}",future.get());
}
newFixedThreadPool.shutdown();
}
运行结果:

就是因为多个线程并发使用SimpleDateFormat,因此出现了异常,那么JDK8之前我们是如何保证SimpleDateFormat线程安全的呢?
在JDK8之前,我们使用ThreadLocal锁定SimpleDateFormat,来保证线程安全:
首先,创建一个使用TreadLocal锁定的SimpleDateFormat
public class DateFormatThreadLocal {
private static final ThreadLocal<DateFormat> df = new ThreadLocal<DateFormat>(){
protected DateFormat initialValue() {
return new SimpleDateFormat("yyyyMMdd");
}
};
public static Date convert(String source) throws Exception{
return df.get().parse(source);
}
}
其次,在做时间格式化时,使用该类中锁定的SimpleDateFormat对象
@Test
public void test2() throws Exception{
//全新的时间API 都不是线程安全的
DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern("yyyyMMdd"); Callable<LocalDate> call = new Callable<LocalDate>() {
@Override
public LocalDate call() throws Exception {
return LocalDate.parse("",dateTimeFormatter);
}
}; ExecutorService newFixedThreadPool = Executors.newFixedThreadPool(); List<Future<LocalDate>> list = new ArrayList<>(); for(int i=;i<;i++){
list.add(newFixedThreadPool.submit(call));
} for (Future<LocalDate> future : list){
log.info("============={}",future.get());
} newFixedThreadPool.shutdown();
}
运行结果:

可以发现,程序运行正常,均做了格式化
那么在JDK8中,提供了全新的时间类,这些类都是线程安全的,可以在多线程并发时使用,无需担心线程安全问题,无需创建ThreadLocal锁定的格式化类
首先,介绍JDK8中新增的时间类 :LocalDate LocalTime LocalDateTime
1、时间对象创建
(1)可以使用now创建时间类,时间为当前时间
(2)可以使用of创建指定时间
//创建日期
//LocalDate LocalTime LocalDateTime
LocalDateTime ld = LocalDateTime.now();
log.info("LocalDateTime.now()=================={}", ld); LocalDateTime ld1 = LocalDateTime.of(,,,,,);
log.info("LocalDateTime=================={}", ld1);
2、日期运算:JDK8提供了plus*方法可以对日期进行运算操作
//日期运算 plus 加日期
LocalDateTime ld = LocalDateTime.now();
log.info("plusDays=================={}", ld.plusDays());
log.info("plusHours=================={}", ld.plusHours());
3、get返回值:JDK8提供了get*方法可以返回年份、月份、日期等
//get 返回值
LocalDateTime ld = LocalDateTime.now();
log.info("getMonthValue=================={}", ld.getMonthValue());
log.info("getDayOfMonth=================={}", ld.getDayOfMonth());
4、时间戳操作
上面提到的LocalDate、LocalTime、LocalDateTime输出的都是指定的时间格式,但是如果我们需要使用时间戳,就需要其他的方式处理,对于时间戳的操作如下:
(1)直接获取时间戳,此时获取的时间戳为UTC时间,即世界协调时间(世界标准时间,以北京时间为例,由于北京时间采取的是东八区时间,因此是标准时间+8个小时,即为北京时间)
(2)获取指定时区的时间戳
(3)转换为时间戳
(4)指定时间戳起始时间(1970-01-01 00:00:00)后多少时间的时间戳
//时间戳
Instant instant = Instant.now();//默认是UTC时间(世界协调时间)
log.info("时间戳===={}", instant);
//转换为东八区时间(北京时间)
OffsetDateTime offsetDateTime = instant.atOffset(ZoneOffset.ofHours());
log.info("转换为东八区时间(北京时间)=={}", offsetDateTime);
//转换为时间戳
long toEpochMilli = instant.toEpochMilli();
log.info("时间戳=={}", toEpochMilli); //改变时间 距离时间戳的时间 1970-01-01 00:00:00
Instant instant1 = Instant.ofEpochSecond(*);
log.info("距离时间戳的时间=={}", instant1);
运行结果:

通过运行结果可以发现,时间戳和根据时区获取的时间戳输出时,仍然是格式化过的时间格式,且是由使用东八区的时间时,才与我电脑上的北京时间一致,同时输出的时间上有+08:00的输出;
另外,对于最后一个距离时间戳的时间,我们设置了60*23,即设置了23分钟,最后输出时间为:1970-01-01T00:23:00Z
5、计算时间差
(1)Duration:获取两个时间的时间差
(2)Period:获取两个日期的间隔
//计算时间差
//Duration:获取两个时间的时间差
//Period:获取两个日期的间隔 Instant instant2 = Instant.now();
Thread.sleep();
Instant instant3 = Instant.now();
Duration duration = Duration.between(instant2,instant3); log.info("duration.getSeconds()============={}", duration.getSeconds());
log.info("duration.toMillis()============={}", duration.toMillis());
log.info("duration.toHours()============={}", duration.toHours()); LocalDate localDate = LocalDate.of(,,);
LocalDate localDate1 = LocalDate.now();
log.info("LocalDate.now()============={}", localDate1);
Period period = Period.between(localDate,localDate1);
log.info("period============={}", period);
log.info("period.getYears()============={}", period.getYears());
log.info("period.getMonths()============={}", period.getMonths());
log.info("period.getDays()============={}", period.getDays());
运行结果:

(1)我们在代码中,让程序休眠了1秒,因此两个时间相差的秒数为1,毫秒数为1001是因为程序运行使用了1毫秒
(2)计算日期差时,我们使用了指定日期2018.10.6和当前日期(2020.6.2)做比较,输出为P1Y7M27D,表示差了1年7个月27天,然后使用get*获取时,获取的即这个输出的年、月、天
6、时间矫正器
(1)可以获取指定该年中的天数
(2)获取下一个指定的日期(下一个周日,JDK已提供)
(3)获取下一个工作日/结婚纪念日等(JDK未提供)
/**
* 时间矫正器
*/
@Test
public void test4(){
//TemporalAdjuster:时间矫正器
LocalDate localDate2 = LocalDate.now();
log.info("LocalDate.now()============={}", localDate2);
//指定时间
LocalDate localDate3 = localDate2.withDayOfYear();
log.info("withDayOfYear============={}", localDate3);
//获取下一个周日
LocalDate localDate4 = localDate2.with(TemporalAdjusters.next(DayOfWeek.SUNDAY));
log.info("TemporalAdjuster============={}", localDate4);
//自定义 下一个工作日
LocalDate localDate5 = localDate2.with((x)->{
LocalDate localDate6 = (LocalDate) x;
DayOfWeek dayOfWeek = localDate6.getDayOfWeek();
switch (dayOfWeek){
case FRIDAY:
return localDate6.plusDays();
case TUESDAY:
return localDate6.plusDays();
default:
return localDate6.plusDays();
}
});
log.info("下一个工作日============={}", localDate5);
}
输出结果:
-- ::26.457 INFO --- [ main] com.example.jdk8demo.Jdk8demoTest2 : LocalDate.now()=============--
-- ::26.461 INFO --- [ main] com.example.jdk8demo.Jdk8demoTest2 : withDayOfYear=============--
-- ::26.463 INFO --- [ main] com.example.jdk8demo.Jdk8demoTest2 : TemporalAdjuster=============--
-- ::26.464 INFO --- [ main] com.example.jdk8demo.Jdk8demoTest2 : 下一个工作日=============--
7、时间格式化
/**
* 时间日期格式化
*
*/
@Test
public void test6(){
DateTimeFormatter df = DateTimeFormatter.ISO_DATE_TIME;
LocalDateTime ld = LocalDateTime.now();
String date = ld.format(df);
log.info("format==================={}",date); //自定义
DateTimeFormatter df1 = DateTimeFormatter.ofPattern("yyyy年MM月dd日 HH时mm分ss秒");
date = ld.format(df1);
log.info("format==================={}",date);
//解析回原格式,此处要注意一点, HH时mm分ss秒大小写一定注意,错了就会反解析失败
LocalDateTime localDateTime = ld.parse(date,df1);
log.info("parse==================={}",localDateTime);
}
8、时区操作
(1)获取所有时区
(2)根据时区获取时间
(3)计算时区的时间差
/**
* 时区操作
* ZoneDate ZoneTime ZoneDateTime
*/
@Test
public void test7(){
//获取所有时区
Set<String> set = ZoneId.getAvailableZoneIds();
log.info("ZoneId.getAvailableZoneIds()================={}",JSON.toJSONString(set));
//根据时区获取时间
LocalDateTime localDateTime = LocalDateTime.now(ZoneId.of("Pacific/Fiji"));
LocalDateTime localDateTime1 = LocalDateTime.now(ZoneId.of("Asia/Shanghai"));
log.info("根据时区获取日期=========Pacific/Fiji==={}==========Asia/Shanghai==={}",localDateTime, localDateTime1);
//获取时差
ZonedDateTime zonedDateTime = localDateTime1.atZone(ZoneId.of("Pacific/Fiji"));
log.info("Shanghai与Fiji的时差========{}",zonedDateTime);
}
JDK8--09:全新的时间API的更多相关文章
- JDK8 新增的日期时间API
背景 JDK8中增加了一套全新的日期时间API,这里进行总结下,方便查询使用. 新的时间及日期API位于 java.time 包中,下面是一些关键类. Instant:代表的是时间戳. LocalDa ...
- JDK8中的新时间API:Duration Period和ChronoUnit介绍
目录 简介 Duration Period ChronoUnit 简介 在JDK8中,引入了三个非常有用的时间相关的API:Duration,Period和ChronoUnit. 他们都是用来对时间进 ...
- JDK8中新日期时间API
它们面临的问题是:可变性:像日期和时间这样的类应该是不可变的.偏移性:Date中的年份是从1900开始的,而月份都从0开始.格式化:格式化只对Date有用,Calendar则不行.此外,它们也不是线程 ...
- java 数据结构(三):java常用类 三 日期时间API
JDK 8之前日期时间API 1.获取系统当前时间:System类中的currentTimeMillis()long time = System.currentTimeMillis();//返回当前时 ...
- Java 常用类-程序员头大的日期时间API
第二节.日期时间API 一.JDK8之前日期时间API 1.1 java.lang.System类 System类提供的public static long currentTimeMillis()用来 ...
- Day029 JDK8中新日期和时间API (二)
# JDK8中新日期和时间API (二) Instant介绍 Instant:时间线上的一个瞬时点. 这可能被用来记录应用程序中的事件时间 戳. 在处理时间和日期的时候,我们通常会想到年,月,日,时, ...
- Day029 JDK8中新日期和时间API (四)
JDK8中新日期和时间API 其他的一些API ZoneId:该类中包含了所有的时区信息,一个时区的ID,如 Europe/Paris ZonedDateTime:一个在ISO-8601日历系统时区的 ...
- JDK8中的时间API
在Java 1.0中,对日期和时间的支持只能依赖java.util.Date类.正如类名所表达的,这个类无法表示日期,只能以毫秒的精度表示时间.更糟糕的是它的易用性,由于某些原因未知的设计决策,这个类 ...
- Java8 时间 API
前言 Java8 中最为人津津乐道的新改变恐怕当属函数式 API 的加入.但实际上,Java8 所加入的新功能远不止这个. 本文将基于<Java SE8 for the Really Impat ...
随机推荐
- Spring boot Sample 008之spring-boot-logback
一.环境 1.1.Idea 2020.1 1.2.JDK 1.8 二.目的 spring boot 整合log4j2 二.步骤 2.1.点击File -> New Project -> S ...
- Rocket - devices - CLINT
https://mp.weixin.qq.com/s/4LfZZDKCTQhiKIUjvbDKEg 简单介绍CLINT的实现. 1. 概述 CLINT即是Core Local Interrupter的 ...
- [C#.NET拾遗补漏]01:字符串操作
字符串操作在任意编程语言的日常编程中都随处可见,今天来汇总一下 C# 中关于字符串的一些你可能遗忘或遗漏的知识点. 逐字字符串 在普通字符串中,反斜杠字符是转义字符.而在逐字字符串(Verbatim ...
- 如何选出适合自己的管理Helm Chart的最佳方式?
本文转载自Rancher Labs 无论你喜欢与否,你都不得不承认Helm是管理Kubernetes应用程序独一无二的工具,你甚至可以通过不同的方式使用它. 在Helm的使用过程中,我们注意到有几个问 ...
- 可以Postman,也可以cURL.进来领略下cURL的独门绝技
文章已经收录在 Github.com/niumoo/JavaNotes ,更有 Java 程序员所需要掌握的核心知识,欢迎Star和指教. 欢迎关注我的公众号,文章每周更新. cURL 是一个开源免费 ...
- EntityFramework数据持久化 Linq介绍
一.LINQ概述与查询语法 二.LINQ方法语法基础(重点) 三.LINQ聚合操作与元素操作(重点) 四.数据类型转换(重点) 一.LINQ概述与查询语法 1.LINQ(Language Integr ...
- Java 第十一届 蓝桥杯 省模拟赛 计算机存储中有多少字节
计算机存储中有多少字节 题目 问题描述 在计算机存储中,12.5MB是多少字节? 答案提交 这是一道结果填空的题,你只需要算出结果后提交即可.本题的结果为一个整数,在提交答案时只填写这个整数,填写多余 ...
- Java实现 LeetCode 887 鸡蛋掉落(动态规划,谷歌面试题,蓝桥杯真题)
887. 鸡蛋掉落 你将获得 K 个鸡蛋,并可以使用一栋从 1 到 N 共有 N 层楼的建筑. 每个蛋的功能都是一样的,如果一个蛋碎了,你就不能再把它掉下去. 你知道存在楼层 F ,满足 0 < ...
- Java实现 蓝桥杯VIP 算法训练 连续正整数的和
问题描述 78这个数可以表示为连续正整数的和,1+2+3-+12,18+19+20+21,25+26+27. 输入一个正整数 n(<=10000) 输出 m 行(n有m种表示法),每行是两个正整 ...
- opencl(3)程序、内核
1:程序 1)从上下文中创建程序 cl_program clCreateProgramWithSource( cl_context context, //上下文 cl_uint count, //文本 ...