SimpleDateFormat 的线程安全问题与解决方案
来源:互联网 发布:moonlight软件 编辑:程序博客网 时间:2024/05/21 16:35
SimpleDateFormat 的线程安全问题
SimpleDateFormat 是一个以国别敏感的方式格式化和分析数据的具体类。 它允许格式化 (date -> text)、语法分析 (text -> date)和标准化。
但是 SimpleDateFormat 并不是一个线程安全的类,在多线程并发访问下会出现问题。通过以下代码进行检验,
多次运行,便会出现异常错误:
线程访问的情况大致如下图:
SimpleDateFormat 类内部有一个 Calendar 对象引用,它用来储存和这个 SimpleDateFormat 相关的日期信息,例如sdf.parse(dateStr), sdf.format(date) 诸如此类的方法参数传入的日期相关String, Date等等, 都是交给 Calendar 引用来储存的.这样就会导致一个问题,如果你的 SimpleDateFormat 是个static 的, 那么多个thread 之间就会共享这个SimpleDateFormat , 同时也是共享这个Calendar引用,那么就出现时间混乱的情况。
解决方法
(1)第一种方法,也是最简单的解决方案。我们可以把static去掉,这样每个新的线程都会有一个自己的sdf实例,从而避免线程安全的问题。
此时访问的情况如下:
然而,使用这种方法,在高并发的情况下会大量的new sdf以及销毁sdf,这样是非常耗费资源的。
(2)第二种方法,使用 ThreadLocal。
在并发情况下,网站的请求任务与线程执行情况大概可以理解为如下。
例如Tomcat的线程池的最大Thread数为4, 现在需要执行的任务有1000个(理解为有1000个用户点了你的网站的某个功能),而这1000个任务都会用到我们写的日期函数处理类
A) 假如说日期函数处理类使用的是new SimpleDateFormat的方法,那么这里就会有1000次sdf的创建和销毁
B) Java中提供了一种ThreadLocal的解决方案,它的工作方式是,每个线程只会有一个实例,也就是说我们执行完这1000个任务,总共只会实例化4个sdf.
而且,它并不会有多线程的并发问题。因为,单个线程执行任务肯定是顺序的,例如Thread #1负责执行Task #1-#250, 那么他是顺序而执行Task #1-#250,而Thread #2拥有自己的sdf实例,他也是顺序执行任务 Task #251-#500, 以此类推。
这里每个线程都有以 Thread 为key的 Map表,而这个表又以pattern 为key,每一个 pattern 都有一个唯一的 SimpleDateFormat 对象。
我们通过下面代码来测试:
从输出我们可以看出:
1) 1个线程执行这6个任务的时候,这个线程首次使用过的时候会new一个新的sdf,并且以后都一直用这个sdf,而不是每次处理任务都新建一个新的sdf
2) 2个线程执行6个任务的时候也是同理,但是2个线程的sdf是分开的,每个线程都有自己的"yyyy-MM-dd", "yyyy-MM"的sdf,所以他们不会有线程安全安全问题
试想,如果使用的是new的实现方法,那么不管是用1个线程去执行,还是用2个线程去执行这6个任务,都需要new 6个sdf
(3)第三种方式,使用同步代码块(synchronized)或者使用装饰器设计模式包装下 SimpleDateFormat ,使之变得线程安全。
也就是在另一篇文章中介绍的 实例封闭机制。
http://blog.csdn.net/zq602316498/article/details/40143437
(4)第四种方式,使用第三方日期处理函数
比如 JODA 来避免这些问题,你也可以使用 commons-lang 包中的 FastDateFormat 工具类。
参考博文:
http://my.oschina.net/leejun2005/blog/152253
http://www.cnblogs.com/zemliu/archive/2013/08/29/3290585.html
- SimpleDateFormat 的线程安全问题与解决方案
- SimpleDateFormat的线程安全问题与解决方案
- SimpleDateFormat的线程安全问题与解决方案
- SimpleDateFormat 的线程安全问题与解决方案
- SimpleDateFormat的线程安全问题与解决方案
- SimpleDateFormat的线程安全问题与解决方案
- SimpleDateFormat 的线程安全问题与解决方案
- SimpleDateFormat的线程安全问题与解决方案
- SimpleDateFormat 的线程安全问题与解决方案
- SimpleDateFormat线程安全问题解决方案
- SimpleDateFormat线程安全问题原因与解决方案
- SimpleDateFormat 的线程安全问题与 ThreadLocal
- SimpleDateFormat的线程安全问题
- SimpleDateFormat的线程安全问题
- SimpleDateFormat的线程安全问题
- SimpleDateFormat 的线程安全问题
- 关于 SimpleDateFormat 的非线程安全问题及其解决方案
- 关于 SimpleDateFormat 的非线程安全问题及其解决方案
- python绘图
- Android崩溃异常捕获方法
- 习题 4.9 给一个不多于5位的正整数,要求:1. 求出它是几位数;2. 分别输出每一位数字;3. 按逆序输出各位数字,例如原数为321,应输出123。
- 初级shell脚本编写示例
- 用Unity实现《随机生成 Tile Based 地图之——洞穴》中的算法
- SimpleDateFormat 的线程安全问题与解决方案
- Qt中的三种基类
- 数据分析之Pandas(二):索引、过滤 、算术运算、 函数应用和映射
- javaseday13
- hdu 6040 Hints of sd0061
- 技术/设计网站收集
- 同一进程间的线程共享和独立的资源
- Java数据库之删除记录
- 在 overlay 中运行容器