在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的更多相关文章

  1. JDK8 新增的日期时间API

    背景 JDK8中增加了一套全新的日期时间API,这里进行总结下,方便查询使用. 新的时间及日期API位于 java.time 包中,下面是一些关键类. Instant:代表的是时间戳. LocalDa ...

  2. JDK8中的新时间API:Duration Period和ChronoUnit介绍

    目录 简介 Duration Period ChronoUnit 简介 在JDK8中,引入了三个非常有用的时间相关的API:Duration,Period和ChronoUnit. 他们都是用来对时间进 ...

  3. JDK8中新日期时间API

    它们面临的问题是:可变性:像日期和时间这样的类应该是不可变的.偏移性:Date中的年份是从1900开始的,而月份都从0开始.格式化:格式化只对Date有用,Calendar则不行.此外,它们也不是线程 ...

  4. java 数据结构(三):java常用类 三 日期时间API

    JDK 8之前日期时间API 1.获取系统当前时间:System类中的currentTimeMillis()long time = System.currentTimeMillis();//返回当前时 ...

  5. Java 常用类-程序员头大的日期时间API

    第二节.日期时间API 一.JDK8之前日期时间API 1.1 java.lang.System类 System类提供的public static long currentTimeMillis()用来 ...

  6. Day029 JDK8中新日期和时间API (二)

    # JDK8中新日期和时间API (二) Instant介绍 Instant:时间线上的一个瞬时点. 这可能被用来记录应用程序中的事件时间 戳. 在处理时间和日期的时候,我们通常会想到年,月,日,时, ...

  7. Day029 JDK8中新日期和时间API (四)

    JDK8中新日期和时间API 其他的一些API ZoneId:该类中包含了所有的时区信息,一个时区的ID,如 Europe/Paris ZonedDateTime:一个在ISO-8601日历系统时区的 ...

  8. JDK8中的时间API

    在Java 1.0中,对日期和时间的支持只能依赖java.util.Date类.正如类名所表达的,这个类无法表示日期,只能以毫秒的精度表示时间.更糟糕的是它的易用性,由于某些原因未知的设计决策,这个类 ...

  9. Java8 时间 API

    前言 Java8 中最为人津津乐道的新改变恐怕当属函数式 API 的加入.但实际上,Java8 所加入的新功能远不止这个. 本文将基于<Java SE8 for the Really Impat ...

随机推荐

  1. 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 ...

  2. Rocket - devices - CLINT

    https://mp.weixin.qq.com/s/4LfZZDKCTQhiKIUjvbDKEg 简单介绍CLINT的实现. 1. 概述 CLINT即是Core Local Interrupter的 ...

  3. [C#.NET拾遗补漏]01:字符串操作

    字符串操作在任意编程语言的日常编程中都随处可见,今天来汇总一下 C# 中关于字符串的一些你可能遗忘或遗漏的知识点. 逐字字符串 在普通字符串中,反斜杠字符是转义字符.而在逐字字符串(Verbatim ...

  4. 如何选出适合自己的管理Helm Chart的最佳方式?

    本文转载自Rancher Labs 无论你喜欢与否,你都不得不承认Helm是管理Kubernetes应用程序独一无二的工具,你甚至可以通过不同的方式使用它. 在Helm的使用过程中,我们注意到有几个问 ...

  5. 可以Postman,也可以cURL.进来领略下cURL的独门绝技

    文章已经收录在 Github.com/niumoo/JavaNotes ,更有 Java 程序员所需要掌握的核心知识,欢迎Star和指教. 欢迎关注我的公众号,文章每周更新. cURL 是一个开源免费 ...

  6. EntityFramework数据持久化 Linq介绍

    一.LINQ概述与查询语法 二.LINQ方法语法基础(重点) 三.LINQ聚合操作与元素操作(重点) 四.数据类型转换(重点) 一.LINQ概述与查询语法 1.LINQ(Language Integr ...

  7. Java 第十一届 蓝桥杯 省模拟赛 计算机存储中有多少字节

    计算机存储中有多少字节 题目 问题描述 在计算机存储中,12.5MB是多少字节? 答案提交 这是一道结果填空的题,你只需要算出结果后提交即可.本题的结果为一个整数,在提交答案时只填写这个整数,填写多余 ...

  8. Java实现 LeetCode 887 鸡蛋掉落(动态规划,谷歌面试题,蓝桥杯真题)

    887. 鸡蛋掉落 你将获得 K 个鸡蛋,并可以使用一栋从 1 到 N 共有 N 层楼的建筑. 每个蛋的功能都是一样的,如果一个蛋碎了,你就不能再把它掉下去. 你知道存在楼层 F ,满足 0 < ...

  9. Java实现 蓝桥杯VIP 算法训练 连续正整数的和

    问题描述 78这个数可以表示为连续正整数的和,1+2+3-+12,18+19+20+21,25+26+27. 输入一个正整数 n(<=10000) 输出 m 行(n有m种表示法),每行是两个正整 ...

  10. opencl(3)程序、内核

    1:程序 1)从上下文中创建程序 cl_program clCreateProgramWithSource( cl_context context, //上下文 cl_uint count, //文本 ...