SimpleDateFormat 的并发安全问题
-
很多人为了提高性能,会把 private static final SimpleDateFormat simpleDateFormat=new SimpleDateFormat(“yyyy-MM-dd HH:mm:ss”); 作为全局变量来写代码;这样写代码会有并发问题,因为 在调用format 或者parse等方法的时候,该类的对象会用到全局变量,一旦涉及到全局变量问题,那么同一个类的对象去面对并发多线程问题,会互相影响彼此的操作数据,导致可能出现异常情况。
----------------------------------------产生线程安全的原因:---------------------------------------------
在SimpleDateFormat转换日期是通过Calendar对象来操作的,SimpleDateFormat继承DateFormat类,DateFormat类中维护一个Calendar对象,代码如下:
可知SimpleDateFormat维护的用于format和parse方法计算日期-时间的calendar被清空了,如果此时线程A将calendar清空且没有设置新值,线程B也进入parse方法用到了SimpleDateFormat对象中的calendar对象,此时就会产生线程安全问题!
————————————————
解决SimpleDateFormat 的并发问题
- 一:当前比较受欢迎的方法,利用空间换时间,放在threadLocal中,保证了线程的安全,借助threadlocal对象每个线程只创建一个实例
public class ConcurrentDateFormat {
private static final ThreadLocal<SimpleDateFormat> _threadLocal=new ThreadLocal<SimpleDateFormat>(){
@Override
protected SimpleDateFormat initialValue() {
return new SimpleDateFormat(yyyy_mmdd_hhmm);
}
};
/**
* 格式化时间
*/
public static String formatDateTime(Date date){
return _threadLocal.get().format(date);
}
}
- 二:加锁,简单粗暴
public class tests {
private static final SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss");
public synchronized static String DateToStringFormat(Date date) {
return format.format(date);
}
}
- 二:jdk8之后的time包,jdk8之后引入了,java.time.XXX相关的包,是线程安全的类。Date 和 LocalDateTime相互转化相关,
public class tests {
// private static final SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss");
private static final DateTimeFormatter _format = DateTimeFormatter.ofPattern("yyyy-MM-dd hh:mm:ss");
public synchronized static String DateToStringFormat(Date date) {
LocalDateTime time = LocalDateTime.ofInstant(date.toInstant(), ZoneId.systemDefault());
return _format.format(time);
}
}