今天用代码测试工具时,发现我的代码中存在这么一个问题,还不能忽略.
报错代码:
public static final SimpleDateFormat FORMAT = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
工具的提示为:
Not all classes in the standard Java library were written to be thread-safe. Using them in a multi-threaded manner is highly likely to cause data problems or exceptions at runtime.This rule raises an issue when an instance of Calendar, DateFormat, javax.xml.xpath.XPath, or javax.xml.validation.SchemaFactory is marked static.Noncompliant Code Examplepublic class MyClass { static private SimpleDateFormat format = new SimpleDateFormat("HH-mm-ss"); // Noncompliant static private Calendar calendar = Calendar.getInstance(); // NoncompliantCompliant Solutionpublic class MyClass { private SimpleDateFormat format = new SimpleDateFormat("HH-mm-ss"); private Calendar calendar = Calendar.getInstance();
原因是SimpleDateFormat类是线程不安全的。这一点其实在JDK中也有提醒:
JDK原始文档如下:
Synchronization: Date formats are not synchronized. It is recommended to create separate format instances for each thread. If multiple threads access a format concurrently, it must be synchronized externally.
由于本次程序并发程度较高,且时间格式化的使用频率也高,所以不能像以往一样无视这种可能性了。于是可选择的解决方式有:
- 用的时候new,用一次new一次 - - !
- synchronized
- ThreadLocal
- 放弃SimpleDateFormat
最终我选择了放弃SimpleDateFormat,使用Apache里的FastDateFormat:
public static final FastDateFormat FORMAT=FastDateFormat.getInstance("yyyy-MM-dd HH:mm:ss");
搞定,不再报错了。
参考好文:
- 20190605更新
java1.8提供了新的时间处理包java.time。可使用java.time.format.DateTimeFormatter格式化时间,如:
/** * 时间戳数字格式化成指定格式. * * @param milliTimestamp 时间戳 * @param pattern 时间格式 * @return */ public static String format(long milliTimestamp, String pattern) { return DateTimeFormatter.ofPattern(pattern) .format(LocalDateTime.ofInstant(Instant.ofEpochMilli(milliTimestamp), ZoneId.systemDefault())); }