细解SimpleDateFormat

其实,作为一名Java 程序员,我们会经常在编程时候和时间打交道,比如要把某一个时间存储到数据库中去,而我们直接使用

Datedate = newDate(); System.out.println(date);

得到的时间都是这样的

SunJun 07 17:22:52CST 2020

因此,我们经常需要把时间进行格式化处理,然后在进行存储,方便阅读。这个时候我们就会使用到SimpleDateFormat 类,比如使用下面的代码来获取当前时间,并调用SimpleDateFormat 对时间进行格式化:

Date date = newDate(); SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");//设置日期格式String datestr = df.format(date); System.out.println(datestr);

最终输出的时间为

2020-06-07 16:45:58

由于在java 8之前 SimpleDateFormat 是一个比较常用的类,但是我还是在这里要建议开发者不要用 SimpleDateForma。原因有两点:

首先,通过new 一个对象来操作对象,占用内存大,如果每处理一个时间信息的时候,就需要new一个SimpleDateFormat实例对象,然后再丢弃这个对象。大量的对象就这样被创建出来,占用大量的内存和 jvm空间。

那么很多人就会想,既然new 代价太大, 不如我们使用 static 将其设置为共享变量,这样就可以减小频繁创建对象带来的内存开销啦,真的是机智如我。

privatestatic SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");

但是这样操作在并发量非常大的情况下,由于 SimpleDateFormat 是线程不安全的,这也是第二点原因,这个在JDK文档中已经明确表明了SimpleDateFormat不应该用在多线程场景中:

Date formats are not synchronized. It is recommended to create separate format instances for each thread. If multiple threads access a format concurrently, it must be synchronized externally.

在《阿里巴巴 Java 开发手册》中也明确表示不要定义SimpleDateFormat 为static 变量,如果定义static 必须要加锁。

 

 

这背后的原因是由于 SimpleDateFormat 中的format方法在执行过程中,会使用一个成员变量calendar来保存时间。

private StringBuffer format(Date date, StringBuffer toAppendTo, FieldDelegate delegate) {this.calendar.setTime(date); boolean useDateFormatSymbols = this.useDateFormatSymbols();int i = 0; ……

由于在声明SimpleDateFormat的时候,使用的是static定义的。那么这个SimpleDateFormat就是一个共享变量,SimpleDateFormat 中的calendar也就可以被多个线程访问到。

举个例子:假设一个线程A刚执行完calendar.setTime把时间设置成2020-05-07,这个线程还没执行完,线程B又执行了calendar.setTime把时间改成了2020-06-07。这时候线程A继续往下执行,拿到的calendar.getTime得到的时间就是线程B改过之后的。

除了format方法以外,SimpleDateFormat的parse方法也有同样的问题。

所以,不要把SimpleDateFormat作为一个共享变量使用。

那么如何解决这样的问题呢?如果你使用的是Java 8 之前的JDK,那么上面的《阿里巴巴Java 开发手册》已经给出了解决方案,那就是使SimpleDateFormat 变成线程安全的,通过加锁的方式来解决

privatestatic ThreadLocal<DateFormat> threadLocal = new ThreadLocal<DateFormat>() {@Overrideprotected DateFormat initialValue(){returnnew SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); } };

而如果你使用的是Java 8 + 的版本,那么你完全可以抛弃这种线程不安全的时间格式化方法。可以使用DateTimeFormatter代替SimpleDateFormat,这是一个线程安全的格式化工具类。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值