分类
三种操作时间的方法都不是线程安全的
-
SimpleDateFormat类主要用于转换格式,String←→Date;
-
Date类用于获取毫秒数,或者设置Calendar对象的参数,date.getTime()、calendar.setTime(date)
-
Calendar类用于具体操作,比如获取指定年、月的天数,加一年一月等。
线程安全
SimpleDateFormat 为什么是线程不安全的?
SimpleDateFormat类实现了 sdf.parse(dateStr), sdf.format(date)等时间转化的方法,该类内部维护一个Calendar 全局变量
风险就出在Calendar 全局变量上,不同线程访问该变量会造成线程安全问题
如果sdf是个static的,那么多线程之间会共享sdf, 即共享这个Calendar引用,sdf.parse() 方法有如下的调用:
Date parse() {
calendar.clear(); // 清理calendar
... // 执行一些操作, 设置 calendar 的日期什么的
calendar.getTime(); // 获取calendar的时间
}
-
如果 线程A 调用了 sdf.parse(), 并且进行了 calendar.clear()后还未执行calendar.getTime(),线程B又调用了sdf.parse(), 这时候线程B也执行了sdf.clear()方法,这样就导致线程A的的calendar数据被清空了(实际上A,B的同时被清空了).
-
或者当 A 执行了calendar.clear() 后被挂起,这时候B 开始调用sdf.parse()并顺利i结束,这样 A 的 calendar内存储的的date,变成了后来B设置的calendar的date
解决办法
- 可以使用局部变量
- 可以通过ThreadLocal包裹SimpleDateFormat全局变量(会为每个线程创建一个副本,以空间换时间)