前言:
SimpleDateFormat不是线程安全的,而且创建一个实例的开销是非常昂贵,解析字符串时间时频繁创建生命周期短暂的实例导致性能低下。
SimpleDateFormat的javadoc中有这么句话:
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.
翻译一下:
*日期格式是不同步的.
* 建议为每个线程创建独立的格式实例.
* 如果多线程并发访问同一个格式,则必须保持外部同步.
目前很多公司使用的是一下两种方式去创建的日期工具类:
1、在每个方法里面使用new SimpleDateFormat的形式
public static Date newParse(String strDate) throws ParseException { return new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").parse(strDate); }
注:这种方式是能解决线程安全问题,但是产生了性能问题,主要是创建一个 SimpleDateFormat实例的开销比较昂贵,解析字符串时间时频繁创建生命周期短暂的实例导致性能低下。
2、定义一个全局变量
private static final SimpleDateFormat datetimeFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); /** * 格式化日期与时间 */ public static String formatDatetime(long timestamp) { return datetimeFormat.format(new Date(timestamp)); }
注:这种方式是解决了性能问题,但是没有解决线程安全问题,在多线程环境下,容易发生各种意想不到的问题,如:java.lang.NumberFormatException: multiple points或者java.lang.NumberFormatException: For input string: ""等异常。
解决此问题有两种方式:
1、使用Threadlocal
private static final ThreadLocal<SimpleDateFormat> _threadLocal = new ThreadLocal<SimpleDateFormat>() { protected SimpleDateFormat initialValue() { return new SimpleDateFormat(); } }; /** * 格式化日期与时间 */ public static String formatDatetime(long timestamp) { return _threadLocal.get().format(new Date(timestamp)); }
2、使用享元模式
A、创建SimpleDateFormatFactory类
public class SimpleDateFormatFactory { private static final ThreadLocal<Map<String, SimpleDateFormat>> _threadLocal = new ThreadLocal<Map<String, SimpleDateFormat>>() { protected Map<String, SimpleDateFormat> initialValue() { return new HashMap<String, SimpleDateFormat>(); } }; public static SimpleDateFormat getInstance() { return getInstance(DateUtil.DATE_FORMAT); } public static SimpleDateFormat getInstance(String format) { SimpleDateFormat sdf = _threadLocal.get().get(format); if (null == sdf) { sdf = new SimpleDateFormat(format); _threadLocal.get().put(format, sdf); } return sdf; } }
B、DateUtil类
public class DateUtil { /** * 显示日期的格式,yyyy-MM-dd */ public static final String DATE_FORMAT = "yyyy-MM-dd"; /** * DateFormat,格式:yyyy-MM-dd */ private static DateFormat dateFormat; static { dateFormat = SimpleDateFormatFactory.getInstance(DATE_FORMAT); } /** * 格式化日期与时间 */ public static String formatDatetime(long timestamp) { return dateFormat.format(new Date(timestamp)); } }
优点:可以更好的去减少系统中对象的个数和对外部状态相对独立等好处。
相关推荐
目录SimpleDateFormat诡异bug复现SimpleDateFormat诡异bug字符串日期转Date日期(parse)Date日期转String类型(format)SimpleDateFormat出现bug...ThreadLocal注意事项使用ThreadLocal解决SimpleDateFormat线程安全问题总结...
在Java编程语言中,`SimpleDateFormat`类是一个广泛使用的日期时间格式化工具,但它的线程安全性是一个常常被开发者忽视的问题。标题指出的"simpleDateFormat是线程不安全的",意味着在多线程环境下,如果多个线程...
SimpleDateFormat线程不安全的5种解决方案.md
要解决SimpleDateFormat类的线程安全问题,有多种方法可以选择: 1. 使用ThreadLocal: 可以使用ThreadLocal将SimpleDateFormat对象封装在ThreadLocal中,这样每个线程都有自己的SimpleDateFormat对象,从而避免了...
ThreadLocal 是一个线程局部变量,每个线程都拥有自己独立的副本,不会互相影响,从而避免线程安全问题。以下是使用 ThreadLocal 的示例: ```java import java.text.SimpleDateFormat; import java.util.Date; ...
本文将深入探讨`SimpleDateFormat`的线程安全问题及其解决方案。 ### 1. 线程安全问题的原因 `SimpleDateFormat`内部维护了一个`Calendar`对象,用于处理日期和时间的解析与格式化。由于`SimpleDateFormat`不是...
### 关于SimpleDateFormat的非线程安全问题及其解决方案 #### 一、问题介绍 在Java开发过程中,`SimpleDateFormat`是被广泛使用的日期格式化工具类。然而,在多线程环境下,`SimpleDateFormat`存在非线程安全的...
为了解决SimpleDateFormat非线程安全性的问题,可以使用同步代码来避免问题。例如,可以使用synchronized关键字来同步访问SimpleDateFormat实例,也可以使用ThreadLocal变量来保存每个线程的SimpleDateFormat实例。 ...
Java SimpleDateFormat线程安全问题原理详解 Java SimpleDateFormat线程安全问题是Java开发中一个常见的问题。SimpleDateFormat是Java中一个常用的日期时间格式化类,但是它却存在线程安全问题。在多线程环境下,...
为了解决SimpleDateFormat类的线程安全问题,可以使用ThreadLocal类。ThreadLocal类可以使线程绑定到指定的对象,从而避免了线程之间的对象共享问题。在示例代码中,我们使用ThreadLocal类来绑定SimpleDateFormat...
它提供了与`SimpleDateFormat`类似的功能,但避免了线程安全问题。 5. **池化`DateFormat`实例**: 尽管`DateFormat`不是线程安全的,但可以通过池化技术减少创建新实例的开销。创建一个`DateFormat`池,线程在...
Java中的ThreadLocal是解决线程安全问题的一个重要工具,它提供了一种在多线程环境下为每个线程维护独立变量副本的方法,从而避免了共享状态带来的竞态条件和线程安全问题。 线程安全问题通常由全局变量和静态变量...
在Java SE中,传统的日期格式化常常涉及到线程安全问题,特别是当多个线程共享同一实例的`SimpleDateFormat`时。这是因为`SimpleDateFormat`不是线程安全的,它内部使用了可变的状态来处理日期和时间格式化。下面将...
Java在并发环境中使用SimpleDateFormat时,可能会遇到线程安全问题。下面将介绍六种解决方案来解决这个问题。 方法一:使用局部变量 在需要执行格式化的地方都新建SimpleDateFormat实例,使用局部变量来存放...
* 解决线程安全问题可以使用ThreadLocal、Lock、Atomic变量等方式。 五、线程池的重要性 * 线程池可以重用线程,减少线程创建和销毁的开销。 * 线程池可以提高程序的可扩展性和可靠性。 * ThreadPoolExecutor是...
其次,这种同步机制也不能完全解决线程安全问题。 ThreadLocal的解决方案 现在,让我们来看一下使用ThreadLocal的解决方案。在我们的示例代码中,我们使用了ThreadLocal<SimpleDateFormat>来给每个线程单独创建...
然而,`DateFormat`并不是线程安全的,这意味着在多线程环境中直接使用可能会导致数据不一致或者异常。这篇博客文章《Java DateFormat并发实现》探讨了这个问题以及如何在并发环境下正确地使用`DateFormat`。 `...
假设我们有一个非线程安全的变量需要转换为线程安全的变量。通常的做法是使用同步机制将对象封装到同步块中,但这可能会导致性能瓶颈。另一种更高效的方式是使用ThreadLocal,为每个线程持有一个独立的对象副本,...
总结来说,理解Java并发编程中的线程和线程池技术,包括它们的实现方式、生命周期、执行顺序控制、异步计算模型以及线程安全问题,是提高程序性能和可扩展性的关键。通过合理使用线程池,我们可以有效地管理线程,...