作者:王爵nice
链接:https://zhuanlan.zhihu.com/p/28133858
来源:知乎
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

当你开始使用Java操作日期和时间的时候,会有一些棘手。你也许会通过System.currentTimeMillis()
来返回1970年1月1日到今天的毫秒数。或者使用Date类来操作日期;当遇到加减月份、天数的时候
你又需要用到Calendar类;当需要格式化日期的时候需要使用java.text.DateFormat类。
总而言之在Java中操作日期不是很方便,以至于很多开发者不得不使用第三方库,比如: joda-time

现有API存在的问题

  • 线程安全: Date和Calendar不是线程安全的,你需要编写额外的代码处理线程安全问题
  • API设计和易用性: 由于Date和Calendar的设计不当你无法完成日常的日期操作
  • ZonedDate和Time: 你必须编写额外的逻辑处理时区和那些旧的逻辑

好在JSR 310规范中为Java8添加了新的API,
在java.time包中,新的API纠正了过去的缺陷,

新的日期API

  • ZoneId: 时区ID,用来确定Instant和LocalDateTime互相转换的规则
  • Instant: 用来表示时间线上的一个点
  • LocalDate: 表示没有时区的日期, LocalDate是不可变并且线程安全的
  • LocalTime: 表示没有时区的时间, LocalTime是不可变并且线程安全的
  • LocalDateTime: 表示没有时区的日期时间, LocalDateTime是不可变并且线程安全的
  • Clock: 用于访问当前时刻、日期、时间,用到时区
  • Duration: 用秒和纳秒表示时间的数量

最常用的就是LocalDate、LocalTime、LocalDateTime了,从它们的名字就可以看出是操作日期
和时间的。这些类是主要用于当时区不需要显式地指定的上下文。在本章节中我们将讨论最常用的api。

LocalDate

LocalDate代表一个IOS格式(yyyy-MM-dd)的日期,可以存储 生日、纪念日等日期。
获取当前的日期:

  1. LocalDate localDate = LocalDate.now();
  2. System.out.println("localDate: " + localDate);

输出

  1. localDate: 2017-07-20

LocalDate可以指定特定的日期,调用of或parse方法返回该实例:

  1. LocalDate.of(2017, 07, 20);
  2. LocalDate.parse("2017-07-20");

当然它还有一些其他方法,我们一起来看看:

为今天添加一天,也就是获取明天

  1. LocalDate tomorrow = LocalDate.now().plusDays(1);

从今天减去一个月

  1. LocalDate prevMonth = LocalDate.now().minus(1, ChronoUnit.MONTHS);

下面写两个例子,分别解析日期 2017-07-20,获取每周中的星期和每月中的日:

  1. DayOfWeek thursday = LocalDate.parse("2017-07-20").getDayOfWeek();
  2. System.out.println("周四: " + thursday);
  3. int twenty = LocalDate.parse("2017-07-20").getDayOfMonth();
  4. System.out.println("twenty: " + twenty);

试试今年是不是闰年:

  1. boolean leapYear = LocalDate.now().isLeapYear();
  2. System.out.println("是否闰年: " + leapYear);

判断是否在日期之前或之后:

  1. boolean notBefore = LocalDate.parse("2017-07-20")
  2. .isBefore(LocalDate.parse("2017-07-22"));
  3. System.out.println("notBefore: " + notBefore);
  4. boolean isAfter = LocalDate.parse("2017-07-20").isAfter(LocalDate.parse("2017-07-22"));
  5. System.out.println("isAfter: " + isAfter);

获取这个月的第一天:

  1. LocalDate firstDayOfMonth = LocalDate.parse("2017-07-20")
  2. .with(TemporalAdjusters.firstDayOfMonth());
  3. System.out.println("这个月的第一天: " + firstDayOfMonth);
  4. firstDayOfMonth = firstDayOfMonth.withDayOfMonth(1);
  5. System.out.println("这个月的第一天: " + firstDayOfMonth);

判断今天是否是我的生日,例如我的生日是 2009-07-20

  1. LocalDate birthday = LocalDate.of(2009, 07, 20);
  2. MonthDay birthdayMd = MonthDay.of(birthday.getMonth(), birthday.getDayOfMonth());
  3. MonthDay today = MonthDay.from(LocalDate.now());
  4. System.out.println("今天是否是我的生日: " + today.equals(birthdayMd));

LocalTime

LocalTime表示一个时间,而不是日期,下面介绍一下它的使用方法。

获取现在的时间,输出15:01:22.144

  1. LocalTime now = LocalTime.now();
  2. System.out.println("现在的时间: " + now);

将一个字符串时间解析为LocalTime,输出15:02

  1. LocalTime nowTime = LocalTime.parse("15:02");
  2. System.out.println("时间是: " + nowTime);

使用静态方法of创建一个时间

  1. LocalTime nowTime = LocalTime.of(15, 02);
  2. System.out.println("时间是: " + nowTime);

使用解析字符串的方式并添加一小时,输出16:02

  1. LocalTime nextHour = LocalTime.parse("15:02").plus(1, ChronoUnit.HOURS);
  2. System.out.println("下一个小时: " + nextHour);

获取时间的小时、分钟

  1. int hour = LocalTime.parse("15:02").getHour();
  2. System.out.println("小时: " + hour);
  3. int minute = LocalTime.parse("15:02").getMinute();
  4. System.out.println("分钟: " + minute);

我们也可以通过之前类似的API检查一个时间是否在另一个时间之前、之后

  1. boolean isBefore = LocalTime.parse("15:02").isBefore(LocalTime.parse("16:02"));
  2. boolean isAfter = LocalTime.parse("15:02").isAfter(LocalTime.parse("16:02"));
  3. System.out.println("isBefore: " + isBefore);
  4. System.out.println("isAfter: " + isAfter);

输出 isBefore: true, isAfter: false

在LocalTime类中也将每天的开始和结束作为常量供我们使用:

  1. System.out.println(LocalTime.MAX);
  2. System.out.println(LocalTime.MIN);

输出:

  1. 23:59:59.999999999
  2. 00:00

LocalTime就这些了,下面我们来了解一下LocalDateTime

LocalDateTime

LocalDateTime是用来表示日期和时间的,这是一个最常用的类之一。

获取当前的日期和时间:

  1. LocalDateTime now = LocalDateTime.now();
  2. System.out.println("现在: " + now);

输出

  1. 现在: 2017-07-20T15:17:19.926

下面使用静态方法和字符串的方式分别创建 LocalDateTime 对象

  1. LocalDateTime.of(2017, Month.JULY, 20, 15, 18);
  2. LocalDateTime.parse("2017-07-20T15:18:00");
  3. ``
  4. 同时`LocalDateTime`也提供了相关API来对日期和时间进行增减操作:
  5. ```java
  6. LocalDateTime tomorrow = now.plusDays(1);
  7. System.out.println("明天的这个时间: " + tomorrow);
  8. LocalDateTime minusTowHour = now.minusHours(2);
  9. System.out.println("两小时前: " + minusTowHour);

这个类也提供一系列的get方法来获取特定单位:

  1. Month month = now.getMonth();
  2. System.out.println("当前月份: " + month);

日期格式化

在日常开发中我们用到最多的也许就是日期、时间的格式化了,那在Java8种该如何操作呢?

  1. LocalDateTime now = LocalDateTime.now();
  2. DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");
  3. System.out.println("默认格式化: " + now);
  4. System.out.println("自定义格式化: " + now.format(dateTimeFormatter));
  5. LocalDateTime localDateTime = LocalDateTime.parse("2017-07-20 15:27:44", dateTimeFormatter);
  6. System.out.println("字符串转LocalDateTime: " + localDateTime);

也可以使用DateTimeFormatterformat方法将日期、时间格式化为字符串

  1. DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern("yyyy-MM-dd");
  2. String dateString = dateTimeFormatter.format(LocalDate.now());
  3. System.out.println("日期转字符串: " + dateString);

日期周期

Period类用于修改给定日期或获得的两个日期之间的区别。

给初始化的日期添加5天:

  1. LocalDate initialDate = LocalDate.parse("2017-07-20");
  2. LocalDate finalDate = initialDate.plus(Period.ofDays(5));
  3. System.out.println("初始化日期: " + initialDate);
  4. System.out.println("加日期之后: " + finalDate);

周期API中提供给我们可以比较两个日期的差别,像下面这样获取差距天数:

  1. long between = ChronoUnit.DAYS.between(initialDate, finalDate);
  2. System.out.println("差距天数: " + between);

上面的代码会返回5,当然你想获取两个日期相差多少小时也是简单的。

与遗留代码转换

在之前的代码中你可能出现了大量的Date类,如何将它转换为Java8种的时间类呢?

DateInstant互相转换

  1. Date date = Date.from(Instant.now());
  2. Instant instant = date.toInstant();

Date转换为LocalDateTime

  1. LocalDateTime localDateTime = LocalDateTime.from(new Date());
  2. System.out.println(localDateTime);

LocalDateTimeDate

  1. Date date =
  2. Date.from(localDateTime.atZone(ZoneId.systemDefault()).toInstant());

LocalDateDate

  1. Date date =
  2. Date.from(LocalDate.now().atStartOfDay().atZone(ZoneId.systemDefault()).toInstant());

跟上Java8 - 日期和时间实用技巧,转自知乎王爵nice的更多相关文章

  1. 跟上Java8 - 日期和时间实用技巧

    原文出处:王爵nice 当你开始使用Java操作日期和时间的时候,会有一些棘手.你也许会通过System.currentTimeMillis() 来返回1970年1月1日到今天的毫秒数.或者使用Dat ...

  2. Java8 日期和时间实用技巧

    新的日期API ZoneId: 时区ID,用来确定Instant和LocalDateTime互相转换的规则 Instant: 用来表示时间线上的一个点 LocalDate: 表示没有时区的日期, Lo ...

  3. Java 8 – 日期和时间实用技巧

    当你开始使用Java操作日期和时间的时候,会有一些棘手.你也许会通过System.currentTimeMillis() 来返回1970年1月1日到今天的毫秒数.或者使用Date类来操作日期:当遇到加 ...

  4. Java8 日期与时间 API

    在 Java 中,想处理日期和时间时,通常都会选用 java.util.Date 这个类进行处理.不过不知道是设计者在当时没想好还是其它原因,在 Java 1.0 中引入的这个类,大部分的 API 在 ...

  5. Java8 日期、时间操作

    一.简介 在Java8之前,日期时间API一直被开发者诟病,包括:java.util.Date是可变类型,SimpleDateFormat非线程安全等问题.故此,Java8引入了一套全新的日期时间处理 ...

  6. Java8 日期和时间API

    LocalDate.LocalTime.Instant.Duration.Period 1.1使用LocalDate和LocalTime 1.1.1LocalDate的创建方式和相关方法使用示例 @T ...

  7. Java8 日期和时间类

    新的日期和时间API 新的日期和时间类解决了Date和Calendar类出现的问题 浅尝 LocalDate 日期类 LocalDate of = LocalDate.of(2018, 7, 13); ...

  8. [转] Java8 日期/时间(Date Time)API指南

    [From] http://www.importnew.com/14140.html Java 8日期/时间( Date/Time)API是开发人员最受追捧的变化之一,Java从一开始就没有对日期时间 ...

  9. Java8新特性--日期和时间API

    如何正确处理时间 现实生活的世界里,时间是不断向前的,如果向前追溯时间的起点,可能是宇宙出生时,又或是是宇宙出现之前, 但肯定是我们目前无法找到的,我们不知道现在距离时间原点的精确距离.所以我们要表示 ...

随机推荐

  1. 按照vue文档使用JavaScript钩子但是却不能执行动画?

    大家刚入VUE肯定是先去阅读文档, 在 进入/离开 & 列表过渡 这一章节有一小节 ---------  JavaScript钩子 详情见vue文档:  https://cn.vuejs.or ...

  2. Java连接MySQL数据库,并进行增删改查

    1.具体的代码实现 import java.sql.*; public class DatabaseService { /** * Create Connection * * @param dbtyp ...

  3. TMS320F28335项目开发记录1_CCS的使用介绍

    CCS使用介绍 一.前言 本系列文章记录本人实际项目开发时对ti的DSP28335,以及CCS开发环境等的学习与记录,相对于2812来说,28335的资料还是比較少的,只是原理是相通的,28335说白 ...

  4. Selenium webdriver Java 查找元素

    1.简单查找 By ID: WebElement element=driver.findElement(By.id("userId")); By Name:WebElement e ...

  5. iovec结构体定义及使用 (转)

    I/O向量(struct iovec) readv(2)与writev(2)函数都使用一个I/O向量的概念.这是由所包含的文件定义的: #include <sys/uio.h> 头文件定义 ...

  6. 【Python】self的用法扫盲

    在Python中,我们有两个重要的概念:类与实例 例如:我们在现实生活中人就是一个类,实例就是具体到某一个男人(张三.李四等) 1.类:定义人这个类 class People(object): pas ...

  7. Array,Vector,List,Deque的区别与联系【转+改】

    数组 内存连续分配,长度大小固定,内置的最基础的数据结构之一.支持随机访问和随机存储. 该类型数据所占内存空间最小. Vector 是C++ STL中的一个容器.和数组类似,它拥有一段连续的内存空间, ...

  8. Android实现蓝牙耳机连接

    代码地址如下:http://www.demodashi.com/demo/13259.html 前言 讲讲android对于蓝牙耳机连接技术的实现 今天涉及的内容有: 流程讲解 新建广播Bluetoo ...

  9. 基于Opencv自带BP网络的车标简易识别

    代码地址如下:http://www.demodashi.com/demo/12966.html 记得把这几点描述好咯:代码实现过程 + 项目文件结构截图 + 演示效果 1.准备工作 1.1 训练集和测 ...

  10. 循环List<Object>

    List<Object> infoData=ArrayList<>(); for (int i = 0; i < infoData.size(); i++) { Obje ...