前言
在阿里Java开发规约中,有强制性的提到SimpleDateFormat 是线程不安全的类 ,在使用的时候应当注意线程安全问题,如下:
其实之前已经介绍过使用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个线程,去转化不同的时间字符串,然后做打印输出,运行结果:
(运行此代码也有可能出现由线程安全问题引起的异常)
根据“预期结果”,两边的数字应该是相等的,为何这里输出不相等呢?通过DateFormat源码可以查看:
因为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 ThreadLocalthreadLocal = 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来处理时间,这里就不赘述了,可以参考我之前的文章。