【JAVA】JDK-SimpleDataFormat 线程不安全!
【问题】
publicclassProveNotSafe {
staticSimpleDateFormat df = newSimpleDateFormat("dd-MMM-yyyy", Locale.US);
staticString testdata[] = { "01-Jan-1999", "14-Feb-2001", "31-Dec-2007"}; publicstaticvoidmain(String[] args) {
Runnable r[] = newRunnable[testdata.length];
for(inti = 0; i < r.length; i++) {
finalinti2 = i;
r[i] = newRunnable() {
publicvoidrun() {
try{
for(intj = 0; j < 1000; j++) {
String str = testdata[i2];
String str2 = null;
/* synchronized(df) */{
Date d = df.parse(str);
str2 = df.format(d);
System.out.println("i: "+ i2 + "\tj: "+ j + "\tThreadID: "
+ Thread.currentThread().getId() + "\tThreadName: "
+ Thread.currentThread().getName() + "\t"+ str + "\t"+ str2);
}
if(!str.equals(str2)) {
thrownewRuntimeException("date conversion failed after "+ j
+ " iterations. Expected "+ str + " but got "+ str2);
}
}
} catch(ParseException e) {
thrownewRuntimeException("parse failed");
}
}
};
newThread(r[i]).start();
}
}
}
i:
2
j:
0
ThreadID:
10
ThreadName: Thread-
2
31
-Dec-
2007
31
-Dec-
2007
i:
2
j:
1
ThreadID:
10
ThreadName: Thread-
2
31
-Dec-
2007
31
-Dec-
2007
i:
2
j:
2
ThreadID:
10
ThreadName: Thread-
2
31
-Dec-
2007
31
-Dec-
2007
i:
2
j:
3
ThreadID:
10
ThreadName: Thread-
2
31
-Dec-
2007
31
-Dec-
2007
i:
2
j:
4
ThreadID:
10
ThreadName: Thread-
2
31
-Dec-
2007
31
-Dec-
2007
i:
2
j:
5
ThreadID:
10
ThreadName: Thread-
2
31
-Dec-
2007
31
-Dec-
2007
i:
2
j:
6
ThreadID:
10
ThreadName: Thread-
2
31
-Dec-
2007
31
-Dec-
2007
i:
2
j:
7
ThreadID:
10
ThreadName: Thread-
2
31
-Dec-
2007
31
-Dec-
2007
i:
2
j:
8
ThreadID:
10
ThreadName: Thread-
2
31
-Dec-
2007
31
-Dec-
2007
i:
2
j:
9
ThreadID:
10
ThreadName: Thread-
2
31
-Dec-
2007
31
-Dec-
2007
i:
2
j:
10
ThreadID:
10
ThreadName: Thread-
2
31
-Dec-
2007
31
-Dec-
2007
i:
2
j:
11
ThreadID:
10
ThreadName: Thread-
2
31
-Dec-
2007
31
-Dec-
2007
i:
2
j:
12
ThreadID:
10
ThreadName: Thread-
2
31
-Dec-
2007
31
-Dec-
2007
i:
2
j:
13
ThreadID:
10
ThreadName: Thread-
2
31
-Dec-
2007
31
-Dec-
2007
i:
2
j:
14
ThreadID:
10
ThreadName: Thread-
2
31
-Dec-
2007
31
-Dec-
2007
i:
2
j:
15
ThreadID:
10
ThreadName: Thread-
2
31
-Dec-
2007
31
-Dec-
2007
i:
2
j:
16
ThreadID:
10
ThreadName: Thread-
2
31
-Dec-
2007
31
-Dec-
2007
i:
2
j:
17
ThreadID:
10
ThreadName: Thread-
2
31
-Dec-
2007
11
-Jan-
1999
i:
0
j:
0
ThreadID:
8
ThreadName: Thread-
0
01
-Jan-
1999
11
-Jan-
1999
Exception in thread
"Thread-2"
i:
1
j:
0
ThreadID:
9
ThreadName: Thread-
1
14
-Feb-
2001
11
-Jan-
2001
Exception in thread
"Thread-0"
java.lang.RuntimeException: date conversion failed after
0
iterations. Expected
01
-Jan-
1999
but got
11
-Jan-
1999
at test.date.ProveNotSafe$
1
.run(ProveNotSafe.java:
30
)
at java.lang.Thread.run(Thread.java:
619
)
Exception in thread
"Thread-1"
java.lang.RuntimeException: date conversion failed after
0
iterations. Expected
14
-Feb-
2001
but got
11
-Jan-
2001
at test.date.ProveNotSafe$
1
.run(ProveNotSafe.java:
30
)
at java.lang.Thread.run(Thread.java:
619
)
java.lang.RuntimeException: date conversion failed after
17
iterations. Expected
31
-Dec-
2007
but got
11
-Jan-
1999
at test.date.ProveNotSafe$
1
.run(ProveNotSafe.java:
30
)
at java.lang.Thread.run(Thread.java:
619
)
SimpleDateFormat和DateFormat类不是线程安全的。我们之所以忽视线程安全的问题,是因为从SimpleDateFormat和DateFormat类提供给我们的接口上来看,实在让人看不出它与线程安全有何相干。只是在JDK文档的最下面有如下说明:
SimpleDateFormat中的日期格式不是同步的。推荐(建议)为每个线程创建独立的格式实例。如果多个线程同时访问一个格式,则它必须保持外部同步。
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.
下面我们通过看JDK源码来看看为什么SimpleDateFormat和DateFormat类不是线程安全的真正原因:
SimpleDateFormat继承了DateFormat,在DateFormat中定义了一个protected属性的 Calendar类的对象:calendar。只是因为Calendar累的概念复杂,牵扯到时区与本地化等等,Jdk的实现中使用了成员变量来传递参数,这就造成在多线程的时候会出现错误。
// Called from Format after creating a FieldDelegate
private StringBuffer format(Date date, StringBuffer toAppendTo,
FieldDelegate delegate) {
// Convert input date to time field list
calendar.setTime(date); for (int i = 0; i < compiledPattern.length; ) {
int tag = compiledPattern[i] >>> 8;
int count = compiledPattern[i++] & 0xff;
if (count == 255) {
count = compiledPattern[i++] << 16;
count |= compiledPattern[i++];
} switch (tag) {
case TAG_QUOTE_ASCII_CHAR:
toAppendTo.append((char)count);
break; case TAG_QUOTE_CHARS:
toAppendTo.append(compiledPattern, i, count);
i += count;
break; default:
subFormat(tag, count, delegate, toAppendTo);
break;
}
}
return toAppendTo;
}
线程1调用format方法,改变了calendar这个字段。
中断来了。
线程2开始执行,它也改变了calendar。
又中断了。
线程1回来了,此时,calendar已然不是它所设的值,而是走上了线程2设计的道路。如果多个线程同时争抢calendar对象,则会出现各种问题,时间不对,线程挂死等等。
分析一下format的实现,我们不难发现,用到成员变量calendar,唯一的好处,就是在调用subFormat时,少了一个参数,却带来了这许多的问题。其实,只要在这里用一个局部变量,一路传递下去,所有问题都将迎刃而解。
这个问题背后隐藏着一个更为重要的问题--无状态:无状态方法的好处之一,就是它在各种环境下,都可以安全的调用。衡量一个方法是否是有状态的,就看它是否改动了其它的东西,比如全局变量,比如实例的字段。format方法在运行过程中改动了SimpleDateFormat的calendar字段,所以,它是有状态的
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
2.同步代码块 synchronized(code)
privatestatic SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); publicstatic String formatDate(Date date)throws ParseException{
synchronized(sdf){
return sdf.format(date);
}
} publicstatic Date parse(String strDate) throws ParseException{
synchronized(sdf){
return sdf.parse(strDate);
}
}
3.使用ThreadLocal:
privatestatic ThreadLocal<SimpleDateFormate> threadLocal = new ThreadLocal<SimpleDateFormate>();
【JAVA】JDK-SimpleDataFormat 线程不安全!的更多相关文章
- 【转】关于Java的Daemon线程的理解
原文地址:http://www.cnblogs.com/ChrisWang/archive/2009/11/28/1612815.html 关于Java的Daemon线程的理解 网上对Java的Dae ...
- Java并发3-多线程面试题
1) 什么是线程? 线程是操作系统能够进行运算调度的最小单位,它被包含在进程之中,是进程中的实际运作单位.程序员可以通过它进行多处理器编程,你可以使用多线程对运算密集型任务提速. 2) 线程和进程有什 ...
- Java中的线程Thread总结
首先来看一张图,下面这张图很清晰的说明了线程的状态与Thread中的各个方法之间的关系,很经典的! 在Java中创建线程有两种方法:使用Thread类和使用Runnable接口. 要注意的是Threa ...
- 多线程(七)JDK原生线程池
如同数据库连接一样,线程的创建.切换和销毁同样会耗费大量的系统资源.为了复用创建好的线程,减少频繁创建线程的次数,提高线程利用率可以引用线程池技术.使用线程池的优势有如下几点: 1.保持 ...
- Java并发之线程中断
前面的几篇文章主要介绍了线程的一些最基本的概念,包括线程的间的冲突及其解决办法,以及线程间的协作机制.本篇主要来学习下Java中对线程中断机制的实现.在我们的程序中经常会有一些不达到目的不会退出的线程 ...
- Java四种线程池的学习与总结
在Java开发中,有时遇到多线程的开发时,直接使用Thread操作,对程序的性能和维护上都是一个问题,使用Java提供的线程池来操作可以很好的解决问题. 一.new Thread的弊端 执行一个异步任 ...
- Java高并发 -- 线程池
Java高并发 -- 线程池 主要是学习慕课网实战视频<Java并发编程入门与高并发面试>的笔记 在使用线程池后,创建线程变成了从线程池里获得空闲线程,关闭线程变成了将线程归坏给线程池. ...
- Java高并发--线程安全策略
Java高并发--线程安全策略 主要是学习慕课网实战视频<Java并发编程入门与高并发面试>的笔记 不可变对象 发布不可变对象可保证线程安全. 实现不可变对象有哪些要注意的地方?比如JDK ...
- Java多线程系列——线程池简介
什么是线程池? 为了避免系统频繁地创建和销毁线程,我们可以让创建的线程进行复用.用线程时从线程池中获取,用完以后不销毁线程,而是归还给线程池. JDK 对线程池的支持 为了更好的控制多线程,JDK 提 ...
- java 多线程之 线程优先级和守护线程
线程优先级的介绍 java 中的线程优先级的范围是1-10,默认的优先级是5."高优先级线程"会优先于"低优先级线程"执行. java 中有两种线程:用户线程和 ...
随机推荐
- centos6.7设置非root帐户自动登录
1.在/etc/gdm/custom.conf文件中修改并加入以下这段 [daemon]AutomaticLogin=你的用户名AutomaticLoginEnable=True 2.重启 reboo ...
- css3 动画效果 总结 不断完善~~
1.transition 动画过程改变某个css属性的效果 (比如宽高 颜色) 语法 transition: all 所有元素 + ...
- 深入理解javascript原型和闭包(11)——执行上下文栈
继续上文的内容. 执行全局代码时,会产生一个执行上下文环境,每次调用函数都又会产生执行上下文环境.当函数调用完成时,这个上下文环境以及其中的数据都会被消除,再重新回到全局上下文环境.处于活动状态的执行 ...
- [NHibernate]基本配置与测试
目录 写在前面 nhibernate文档 搭建项目 映射文件 持久化类 辅助类 数据库设计与连接配置 测试 总结 写在前面 一年前刚来这家公司,发现项目中使用的ORM是Nhibernate,这个之前确 ...
- Access应用笔记<四>-一个完整的自动化报表搭建过程
距离之前的三篇日志已经很久啦,今天终于完成了一个比较完整的自动化报表搭建过程 基于公司数据保密原则,样板就不放到网上来了,简单说一下背景: 这次access实现的功能包括: 1)为部门整体搭建了一个员 ...
- 开始学红帽的RHCE课堂有2次课了,要记下自己的学习经历
我终于申请成功了博客园的博客了. 红帽课堂已经开始2次了,这里的记录可能不分顺序,每天记录一点自己的学习内容.方便自己以后查询. 已经学了以下内容: 1.访问命令行 使用桌面的访问命令 GNOME 3 ...
- spring常见问题
问题1:提示说:cvc-elt.1: Cannot find the declaration of element 'beans' 解决方法:从网上搜了一些,有的说是因为网络原因访问不到xsd文件,因 ...
- linux c 笔记-2 Hello World & main函数
按照惯例撸一个hello_world.c #include <stdio.h> int main(int argc, char * argv[]) { printf("hello ...
- WebService -- Java 实现之 CXF ( 使用CXF工具生成client 程序)
1. 下载CXF 工具解压到磁盘 2.添加工具bin目录到PATH环境变量 3.创建一个CXF client新项目 4. run -> cmd 到指定目录,并运行工具目录下的批处理 “wadl2 ...
- opendrive
opendrive和其他许多网盘一样.注册拥有5G的免费空间.每天1G的免费外链流量.更重要的是,他能够给你提供一个直接外链!这是国内外许多网盘都没有的.当你上载了一个MP3,你想用直接外链的形式在博 ...