前言:
在开发中,我们都使用过SimpleDateFormat这个类,更为了开发方便,减少SimpleDateFormat对象的创建,我们将它封装在了 DateUtil 工具类中,并将它定义成了全局静态的变量,这样减少了内存开销,那这样究竟行不行呢?jdk官方文档给了我们答案:
它的意思是说SimpleDateFormat中的日期格式是不同步的。推荐(建议)为每个线程创建独立的实例。如果多个线程同时访问一个实例,则它必须保持同步。
下面我们来看下代码:
1.DateUtil类
2.多线程测试
3.输出结果
分析:
由于SimpleDateFormat类是非线程安全的,我们将它定义成了一个静态的全局变量,在内存中它只会有一份,在多线程环境下,当一个线程对它里面的内容做出修改之后,其他线程均可见,从而会造成一些资源冲突.我们看SimpleDateFormat的源码会发现它的父类DateFormat中定义了一个protected修饰的 Calendar类对象.而在format()方法中使用到calendar.setTime(date),之后,calendar还会用到subFormat方法里,这样在多线程一起访问时就会出现错误.
想象一下
在多线程环境下,有两个线程持有了同一个SimpleDateFormat的实例,分别调用format方法,
线程1调用format方法,改变了calendar. 中断,
线程2执行, 它也改变了calendar. 又中断了
线程1回来了, 此时,calendar已经不是它所设的值,而是变成线程2设置的. 如果多个线程同时争抢calendar对象,则会出现各种问题,时间不对,线程挂死等等
解决方法:
1.改成创建局部变量的方式(局部变量在栈中,每个线程都有一个栈)
2.加同步锁(每次只会有一个线程调用)
3.使用ThreadLocal
在使用中3种方法各有优劣。