还在用SimpleDateFormat?Java8都发布N年了,转LocalDateTime吧
前言
Java8发布,已有数年之久,但是发现很多人都还是坚持着用SimpleDateFormat和Date进行时间操作。SimpleDateFormat这个类不是线程安全的,在使用的时候稍不注意,就会产生致命的问题。Date这个类,是可以重新设置时间的,这对于一些类内部的属性来说,是非常不安全的。
SimpleDateFormat是线程不安全的类
在阿里巴巴规约手册里,强制规定SimpleDateFormat是线程不安全的类,当定义为静态变量时,必须加锁处理。忽略线程安全问题,正是大多数Java初学者在进行时间转化时容易踩坑的点。
Date属性可以重新设置时间
比如有User.java如下:
public class User {
private String username;
private Date birthday;
public User(String username, Date birthday) {
this.username = username;
this.birthday = birthday;
}
public String getUsername() {
return username;
}
public void setUsername(String username) {
this.username = username;
}
public Date getBirthday() {
return birthday;
}
public void setBirthday(Date birthday) {
this.birthday = birthday;
}
}
我们实例化该User
public class Main {
public static void main(String[] args) {
User user = new User("happyjava", new Date());
}
}
这当然没什么问题,但是我可以通过user.getBirthday()方法获取到birthday的引用,从而修改直接修改birthday的值。如下:
public static void main(String[] args) {
User user = new User("happyjava", new Date());
System.out.println(user.getBirthday());
Date birthday = user.getBirthday();
birthday.setTime(11111111L);
System.out.println(user.getBirthday());
}
输出结果如下:
这里可以看到,user对象的birthday属性被修改掉了。这也是Date对象的弊端所在,我们可以通过改写getter方法,使它返回一个新的Date对象即可解决,如下:
public Date getBirthday() {
// return birthday;
return new Date(birthday.getTime());
}
切记这里是不可以用clone方法来生成返回一个新的Date对象的,因为Date类可以被继承,你不能确定调用者是否给birthday设置了一个Date的子类。
Java8提供的新的时间类库LocalDateTime
Java8提供了LocalDateTime来替代传统的Date来处理时间,下面,我们就来探讨下这个类库的使用方法吧。
1.获取当前时间
可以通过 LocalDateTime localDateTime = LocalDateTime.now();方法来获取当前时间,测试如下:
@Test
public void testNow() {
LocalDateTime localDateTime = LocalDateTime.now();
System.out.println(localDateTime);
}
输出结果
2019-05-06T22:25:07.309
2.根据时间戳初始化时间
@Test
public void testNewFromTimestamp() {
Instant instant = Instant.ofEpochMilli(System.currentTimeMillis());
LocalDateTime dateTime = LocalDateTime.ofInstant(instant, ZoneId.of("+8"));
System.out.println(dateTime);
}
这里的+8意思是东八区,下同。
输出结果:
2019-05-06T22:27:34.567
3.根据字符串获取时间
可以使用LocalDateTime.parse方法对字符串进行转化成时间,如果不传pattern,默认是2019-05-06T11:16:12.361格式。
@Test
public void testNewFromString() {
// 1.默认格式 2019-05-06T11:16:12.361
String dateStr = "2019-05-06T11:16:12.361";
LocalDateTime localDateTime = LocalDateTime.parse(dateStr);
System.out.println(localDateTime);
// 2. 自定义格式
String pattern = "yyyy-MM-dd HH:mm:ss";
dateStr = "2019-01-01 12:12:12";
localDateTime = LocalDateTime.parse(dateStr, DateTimeFormatter.ofPattern(pattern));
System.out.println(localDateTime);
}
输出结果:
2019-05-06T11:16:12.361
2019-01-01T12:12:12
4.时间转化成字符串
可以通过DateTimeFormatter的format方法,将LocalDateTime转化成字符串。
@Test
public void testToString() {
LocalDateTime now = LocalDateTime.now(ZoneId.of("+8"));
String pattern = "yyyy-MM-dd HH:mm:ss";
DateTimeFormatter formatter = DateTimeFormatter.ofPattern(pattern);
String format = formatter.format(now);
System.out.println(format);
}
输出结果:
2019-05-06 22:33:03
5.LocalDateTime转时间戳
@Test
public void testDateToTimeMillis() {
LocalDateTime dateTime = LocalDateTime.now();
long epochMilli = dateTime.toInstant(ZoneOffset.of("+8")).toEpochMilli();
System.out.println(epochMilli);
}
输出结果:
1557153504304
总结
因为DateTimeFormatter是线程安全的,所以在实际使用LocalDateTime的时候,可以把DateTimeFormatter定义成静态常量的方式进行使用。以上列举了比较常用的时间操作,LocalDateTime还可以做很多事情,这个就让读者自行去挖掘吧。我自己封装了个LocalDateTime工具类,只做过简单的自测,大家可以参考一下:
package happy.localdatetime;
import java.time.Instant;
import java.time.LocalDateTime;
import java.time.ZoneOffset;
import java.time.format.DateTimeFormatter;
/**
* @author Happy
*/
public class DateTimeUtils {
private DateTimeUtils() {
}
private final static String COMMON_PATTERN = "yyyy-MM-dd HH:mm:ss";
private final static DateTimeFormatter COMMON_FORMATTER = DateTimeFormatter.ofPattern(COMMON_PATTERN);
private final static ZoneOffset DEFAULT_ZONE_OFFSET = ZoneOffset.of("+8");
/**
* 默认 yyyy-MM-dd HH:mm:ss 格式
*/
public static String dateToString(LocalDateTime dateTime) {
assert dateTime != null;
return COMMON_FORMATTER.format(dateTime);
}
/**
* 默认 yyyy-MM-dd HH:mm:ss 格式
*/
public static LocalDateTime stringToDate(String dateStr) {
assert dateStr != null;
return LocalDateTime.parse(dateStr, COMMON_FORMATTER);
}
public static String dateToString(LocalDateTime dateTime, DateTimeFormatter formatter) {
assert dateTime != null;
return formatter.format(dateTime);
}
public static LocalDateTime stringToDate(String dateStr, DateTimeFormatter formatter) {
assert dateStr != null;
return LocalDateTime.parse(dateStr, formatter);
}
public static long dateToTimeMillis(LocalDateTime dateTime) {
assert dateTime != null;
return dateTime.toInstant(DEFAULT_ZONE_OFFSET).toEpochMilli();
}
public static LocalDateTime timeMillisToDate(long timeMillis) {
Instant instant = Instant.ofEpochMilli(timeMillis);
return LocalDateTime.ofInstant(instant, DEFAULT_ZONE_OFFSET);
}
public static void main(String[] args) {
String s = dateToString(LocalDateTime.now());
System.out.println(s);
System.out.println();
String dateStr = "2019-01-01 12:12:12";
LocalDateTime localDateTime = stringToDate(dateStr);
System.out.println(localDateTime);
System.out.println();
System.out.println(dateToTimeMillis(localDateTime));
System.out.println();
System.out.println(timeMillisToDate(System.currentTimeMillis()));
}
}
还在用SimpleDateFormat?Java8都发布N年了,转LocalDateTime吧的更多相关文章
- CentOS 8 都发布了,你还不会用 nftables?
原文链接:CentOS 8 都发布了,你还不会用 nftables? 如果你没有生活在上个世纪,并且是云计算或相关领域的一名搬砖者,那你应该听说最近 CentOS 8 官方正式版已经发布了,CentO ...
- [转帖]CentOS 8 都发布了,你还不会用 nftables?
CentOS 8 都发布了,你还不会用 nftables? https://www.cnblogs.com/ryanyangcs/p/11611730.html 改天学习一下 原文链接:CentOS ...
- 还在用SimpleDateFormat格式化时间?小心经理锤你
还在用SimpleDateFormat格式化时间?小心经理锤你 场景 本来开开心心的周末时光,线上突然就疯狂报错,以为程序炸了,截停日志,发现是就是类似下述一段错误 java.lang.NumberF ...
- java8新特性LocalDate、LocalTime、LocalDateTime的学习
以前操作时间都是使用SimpleDateFormat类改变Date的时间格式,使用Calendar类操作时间.但是SimpleDateFormat是线程不安全的,源码如下: private Strin ...
- JDK10都发布了,nio你了解多少?
前言 只有光头才能变强 回顾前面: 给女朋友讲解什么是代理模式 包装模式就是这么简单啦 本来我预想是先来回顾一下传统的IO模式的,将传统的IO模式的相关类理清楚(因为IO的类很多). 但是,发现在整理 ...
- 还在使用SimpleDateFormat?
阅读本文大概需要 3.2 分钟. 前言 日常开发中,我们经常需要使用时间相关类,想必大家对SimpleDateFormat并不陌生.主要是用它进行时间的格式化输出和解析,挺方便快捷的,但是Simple ...
- 什么,你还使用 webpack?别人都在用 vite 搭建项目了
一.vite 到底是干嘛的? vite 实际上就是一个面向现代浏览器,基于 ES module 实现了一个更轻快的项目构建打包工具. vite 是法语中轻快的意思. vite 的特点: 1.轻快的冷服 ...
- 还在使用SimpleDateFormat?你的项目崩没?
如果是在多线程环境下,就需要注意了. 要点: 1.加Synchronized同步: 2.使用ThreadLocal: 3.jdk8使用DateTimeFormatter替代SimpleDateForm ...
- 从代码都发布遇到的问题总结(C#调用非托管dll文件,部署项目) 转
http://www.cnblogs.com/Purple_Xiapei/archive/2012/06/30/2570928.html
随机推荐
- HBase的完全分布式搭建
一.HBase的安装模式 ①单机安装:不依赖于Hadoop的HDFS,配置完即可使用,好处是便于测试,坏处是不具备分布式数据存储的能力. ②伪分布式安装:单台主机模拟真实环境. ③完全分布式安装:多台 ...
- pycharm新建Django时,遇到的坑,安装index包失败
https://blog.csdn.net/li93675/article/details/89418097 如果在pycharm中导入django包 ,只对当前项目有效,建议使用命令pip inst ...
- Sudo临时提权配置
目录 Sudo临时提权配置 参考 Sudo简介 Sudo配置文件 Sudo配置语法 Sudo配置实例 Sudo日志记录 Sudo临时提权配置
- 批处理执行Testng
@echo off set LIB=项目的jar路径 set CLASSPATH=%LIB%\jar\commons-beanutils-.jar;%LIB%\jar\testng.jar;%LIB% ...
- ASP.NET学习记录点滴
1.判读是否是第一次请求,有表单的页面,第一次请求时get请求,而不是post请求,所以可以用来判断请求是否是get,在apsx页面中,有微软封装的属性IsPostBack来判断是否是get还是pos ...
- Java Https双向验证
CA: Certificate Authority,证书颁发机构 CA证书:证书颁发机构颁发的数字证书 参考资料 CA证书和TLS介绍 HTTPS原理和CA证书申请(满满的干货) 单向 / 双向认证 ...
- ElasticSearch学习记录 - 命令示例
GET /searchfilmcomments/searchfilmcomments/_search { "query": { "match_all": {} ...
- [蓝桥杯2017初赛]k倍区间 前缀和
题目描述 给定一个长度为N的数列,A1, A2, ... AN. 如果其中一段连续的子序列Ai, Ai+1, ... Aj(i <= j)之和是K的倍数,我们就称这个区间[i, j]是K倍区间. ...
- JAVA 开学测试
package StudentScore; public class ScoreInformation { String stunumber; //学号 String name; //姓名 doubl ...
- 全排列next_permutation()用法和构造函数赋值
全排列next_permutation()用法 在头文件aglorithm里 就是1~n数组的现在的字典序到最大的字典序的依次增加.(最多可以是n!种情况) int a[n]; do{ }while( ...