博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
为什么阿里Java规约要求谨慎使用SimpleDateFormat
阅读量:5158 次
发布时间:2019-06-13

本文共 3881 字,大约阅读时间需要 12 分钟。

前言

在阿里Java开发规约中,有强制性的提到SimpleDateFormat 是线程不安全的类 ,在使用的时候应当注意线程安全问题,如下:

16b6b021cf1a0889?w=709&h=299&f=jpeg&s=175941

其实之前已经介绍过使用JDK1.8的DateTimeFormatter 和LocalDateTime来处理时间了,。今天,就来说说SimpleDateFormat的线程安全问题。

SimpleDateFormat是非线程安全的

时间处理,基本所有项目上都是需要使用到的,往往很多初学者会把SimpleDateFormat定义为static类型,然后在进行时间转化的时候没有做加锁处理。如下:

public class Main {    private final static SimpleDateFormat SDFT = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");    public static void main(String[] args) throws ParseException {        System.out.println(SDFT.parse("2019-05-29 12:12:12"));    }}

当然,本代码直接运行是没有问题的。但是,当此SDFT实例应用到多线程环境下的时候,就会出现致命的问题。假设有如下代码:

public class Main {    private static SimpleDateFormat SDFT = new SimpleDateFormat("yyyy-MM-dd");    public static void main(String[] args) {        for (int i = 1; i < 31; i++) {            int ii = i;            new Thread(() -> {                Date date = null;                try {                    String s = "2019-05-" + ii;                    date = SDFT.parse(s);                    System.out.println("" + ii + ":" + date.getDate());                } catch (ParseException e) {                    e.printStackTrace();                }            }).start();        }    }}

此代码的意思是创建30个线程,去转化不同的时间字符串,然后做打印输出,运行结果:

16b6b021cf8fa8f4?w=929&h=739&f=jpeg&s=87113

(运行此代码也有可能出现由线程安全问题引起的异常)

根据“预期结果”,两边的数字应该是相等的,为何这里输出不相等呢?通过DateFormat源码可以查看:

16b6b021d010f0ff?w=855&h=426&f=jpeg&s=251304

16b6b021d027d40d?w=877&h=730&f=jpeg&s=233943

因为SimpleDateFormat定义为了共享的,所以其类里的属性calendar也是多个线程共享的,这就造成了线程安全问题。

解决方案

方案一:加锁处理

如本文例子,可以通过加锁来保证线程安全:

public class Main {    private static SimpleDateFormat SDFT = new SimpleDateFormat("yyyy-MM-dd");    public static void main(String[] args) {        for (int i = 1; i < 31; i++) {            int ii = i;            new Thread(() -> {                Date date = null;                try {                    String s = "2019-05-" + ii;                    synchronized (Main.class) {                        date = SDFT.parse(s);                    }                    System.out.println("" + ii + ":" + date.getDate());                } catch (ParseException e) {                    e.printStackTrace();                }            }).start();        }    }}

输出:

4:43:31:12:229:2928:2827:2726:2630:3025:2523:2321:2120:2022:2218:1824:2419:1917:1716:1614:1415:1512:1213:1310:1011:119:97:76:65:58:8

方案二:每次都创建SimpleDateFormat实例

代码改造如下:

public static void main(String[] args) {        for (int i = 1; i < 31; i++) {            int ii = i;            new Thread(() -> {                Date date = null;                try {                    String s = "2019-05-" + ii;                    date = new SimpleDateFormat("yyyy-MM-dd").parse(s);                    System.out.println("" + ii + ":" + date.getDate());                } catch (ParseException e) {                    e.printStackTrace();                }            }).start();        }    }

每次使用SimpleDateFormat的时候,都去创建一个SimpleDateFormat实例,保证SimpleDateFormat实例不被共享。

方案三:使用LocalThread

这是阿里Java规约里提到的解决方法之一,之所以可以使用LocalThread来解决此问题,代码改造如下:

public class Main {    private static final ThreadLocal
threadLocal = new ThreadLocal
() { @Override protected SimpleDateFormat initialValue() { return new SimpleDateFormat("yyyy-MM-dd"); } }; public static void main(String[] args) { for (int i = 1; i < 31; i++) { int ii = i; new Thread(() -> { Date date = null; try { String s = "2019-05-" + ii; date = threadLocal.get().parse(s); System.out.println("" + ii + ":" + date.getDate()); } catch (ParseException e) { e.printStackTrace(); } }).start(); } }}

运行结果如下:

22:222:224:2415:1517:1716:1629:299:930:303:34:45:512:128:820:2026:2621:2128:2819:1927:2718:181:114:1425:2511:1113:137:76:623:2310:10

解决方法四:使用JDK1.8提供的DateTimeFormatter来处理时间,这里就不赘述了,可以参考我之前的文章。

转载于:https://www.cnblogs.com/happy4java/p/11206096.html

你可能感兴趣的文章
移动APP测试流程
查看>>
SqlServer中创建非聚集索引和非聚集索引
查看>>
单件模式
查看>>
html5 新标签
查看>>
预加载显示图片的艺术
查看>>
MyEclipse使用总结——MyEclipse去除网上复制下来的来代码带有的行号
查看>>
订餐系统之获取淘宝外卖订单
查看>>
会计基础第一节内容概述
查看>>
AE开发中出现无spatial analysis和3D分析等的licence情况
查看>>
嵊州D2T1 “我只是来打个电话”
查看>>
第十周进度条
查看>>
[詹兴致矩阵论习题参考解答]习题2.1
查看>>
切换用户后,/etc/profile的配置不起效
查看>>
ceph<一>安装
查看>>
redis密码管理
查看>>
Json:Restful
查看>>
【iOS】Quartz2D基本图形
查看>>
字符串
查看>>
转:OAuth2 深入介绍
查看>>
hello world``````````
查看>>