SimpleDateFormat做成员或者静态成员多线程安全隐患
转自:http://blog.csdn.net/jeamking/article/details/7183958
有时我们在同一个类中都是使用同一种日期格式,又或者为了减少new SimpleDateFormat次数,自然而然的就会出现如下代码:
private static SimpleDateFormat sdf = newSimpleDateFormat("yyyy-MM-dd HH:mm:ss");
但是这样做在多线程并发下会存在安全隐患。SimpleDateFormat 类并不是线程同步的,JDK中我们可以看到如下描述:
* Date formats are not synchronized.
* Itis recommended to create separate format instances for each thread.
* Ifmultiple threads access a format concurrently, it must be synchronized
*externally.
下面通过一个简单程序来进行测试:
public class DateUtil {
privatestatic SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-ddHH:mm:ss");
publicDate parse(String str) throws ParseException{
returnsdf.parse(str);
}
}
public class Test {
/**
* @param args
* @throws InterruptedException
*/
publicstatic void main(String[] args) throws InterruptedException {
SdfRunnablett = new SdfRunnable();
for(inti=0;i<10;i++){
newThread(tt).start();
}
}
}
输出结果:
Fri Dec 23 12:23:35 CST 2011
Fri Dec 23 12:23:35 CST 2011
Fri Dec 23 12:23:35 CST 2011
Fri Dec 23 12:23:35 CST 2011
Fri Dec 23 12:23:35 CST 2011
Fri Dec 23 12:23:35 CST 2011
Fri Dec 23 12:23:35 CST 2011
Fri Dec 23 12:23:35 CST 2011
Exception in thread "Thread-1"java.lang.NumberFormatException: multiple points
atsun.misc.FloatingDecimal.readJavaFormatString(Unknown Source)
atjava.lang.Double.parseDouble(Unknown Source)
atjava.text.DigitList.getDouble(Unknown Source)
atjava.text.DecimalFormat.parse(Unknown Source)
atjava.text.SimpleDateFormat.subParse(Unknown Source)
atjava.text.SimpleDateFormat.parse(Unknown Source)
atjava.text.DateFormat.parse(Unknown Source)
atDateUtil.parse(DateUtil.java:8)
atSdfRunnable.run(SdfRunnable.java:8)
atjava.lang.Thread.run(Unknown Source)
多运行几次,并不每次都输出这个错误信息,还可能输出如下错误信息:
Fri Dec 23 12:23:35 CST 2011
Fri Dec 23 12:23:35 CST 2011
Fri Dec 23 12:23:35 CST 2011
Fri Dec 23 12:23:35 CST 2011
Fri Dec 23 12:23:35 CST 2011
Wed Sep 23 12:23:35 CST 2201
Fri Dec 23 12:23:35 CST 2011
Exception in thread "Thread-6"java.lang.NumberFormatException: empty String
atsun.misc.FloatingDecimal.readJavaFormatString(Unknown Source)
atjava.lang.Double.parseDouble(Unknown Source)
atjava.text.DigitList.getDouble(Unknown Source)
atjava.text.DecimalFormat.parse(Unknown Source)
atjava.text.SimpleDateFormat.subParse(Unknown Source)
atjava.text.SimpleDateFormat.parse(Unknown Source)
atjava.text.DateFormat.parse(Unknown Source)
atDateUtil.parse(DateUtil.java:8)
atSdfRunnable.run(SdfRunnable.java:8)
atjava.lang.Thread.run(Unknown Source)
Exception in thread "Thread-1"java.lang.NumberFormatException: For input string: ""
atjava.lang.NumberFormatException.forInputString(Unknown Source)
atjava.lang.Long.parseLong(Unknown Source)
atjava.lang.Long.parseLong(Unknown Source)
atjava.text.DigitList.getLong(Unknown Source)
atjava.text.DecimalFormat.parse(Unknown Source)
atjava.text.SimpleDateFormat.subParse(Unknown Source)
atjava.text.SimpleDateFormat.parse(Unknown Source)Fri Dec 23 12:23:35 CST 2011
atjava.text.DateFormat.parse(Unknown Source)
atDateUtil.parse(DateUtil.java:8)
atSdfRunnable.run(SdfRunnable.java:8)
atjava.lang.Thread.run(Unknown Source)
在以上示例中我们把SimpleDateFormat定义为静态成员的。接下来我们把SimpleDateFormat定义为类普通成员。
public class DateUtil {
privateSimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
publicDate parse(String str) throws ParseException{
returnsdf.parse(str);
}
}
再次运行程序,结果如下:
Fri Dec 23 12:23:35 CST 2011
Fri Dec 23 12:23:35 CST 2011
Fri Dec 23 12:23:35 CST 2011
Fri Dec 23 12:23:35 CST 2011
Fri Dec 23 12:23:35 CST 2011
Fri Dec 23 12:23:35 CST 2011
Fri Dec 23 12:23:35 CST 2011
Fri Dec 23 12:23:35 CST 2011
Fri Dec 23 12:23:35 CST 2011
Fri Dec 23 12:23:35 CST 2011
程序运行正常。并没有出现错误信息。那么,是不是意味着SimpleDateFormat作为普通成员就没有安全隐患呢,请看如下示例:
public class SdfRunnable implementsRunnable {
privateSimpleDateFormat sdf;
privateString dateStr;
publicSdfRunnable(SimpleDateFormat sdf,String dateStr){
this.sdf= sdf;
this.dateStr= dateStr;
}
publicvoid run() {
try{
System.out.println(this.sdf.parseObject(this.dateStr));
}catch (ParseException e) {
e.printStackTrace();
}
}
}
public class Test {
/**
* @param args
* @throws InterruptedException
*/
publicstatic void main(String[] args) throws InterruptedException {
SimpleDateFormatsdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
SdfRunnablett = new SdfRunnable(sdf,"2011-12-34 12:23:11");
for(inti=0;i<10;i++){
newThread(tt).start();
}
}
}
运行输出结果如下:
Tue Jan 03 12:23:11 CST 2012
Tue Jan 03 12:23:11 CST 2012
Tue Jan 03 12:23:11 CST 2012
Tue Jan 03 12:23:11 CST 2012
Tue Jan 03 12:23:11 CST 2012
Thu Jan 01 00:00:11 CST 1970
Sun Jan 03 12:23:11 CST 1971
Tue Jan 03 12:23:11 CST 2012
Exception in thread "Thread-9"java.lang.NumberFormatException: multiple points
atsun.misc.FloatingDecimal.readJavaFormatString(Unknown Source)
atjava.lang.Double.parseDouble(Unknown Source)
atjava.text.DigitList.getDouble(Unknown Source)
atjava.text.DecimalFormat.parse(Unknown Source)
atjava.text.SimpleDateFormat.subParse(Unknown Source)
atjava.text.SimpleDateFormat.parse(Unknown Source)
atjava.text.DateFormat.parseObject(Unknown Source)
atjava.text.Format.parseObject(Unknown Source)
atSdfRunnable.run(SdfRunnable.java:14)
atjava.lang.Thread.run(Unknown Source)
由此可见,不管是SimpleDateFormat作为静态成员还是成员,在多线程并发下都会存在安全隐患。所以应该在需要使用的地方再new SimpleDateFormat()。
关于SimpleDateFormat非多线程安全问题,在sun的bug database(http://bugs.sun.com/bugdatabase/view_bug.do?bug_id=4228335)中有更多的描述。
SimpleDateFormat做成员或者静态成员多线程安全隐患的更多相关文章
- C++学习5-面向对象编程基础(构造函数、转换构造、静态数据成员、静态成员函数、友元)
知识点学习 类 const作用 C语言的const限定符的含义为"一个不能改变值的变量",C++的const限定符的含义为"一个有类型描述的常量": const ...
- iOS - 互斥锁&&自旋锁 多线程安全隐患(转载)
一.多线程安全隐患 资源共享 一块资源可能会被多个线程共享,也就是多个线程可能会访问到一块资源 比如多个线程访问同一个对象,同一个变量,同一个文件. 当多线程访问同一块资源的时候,很容易引发数据错乱 ...
- JS OOP -04 JS中的公有成员,私有成员和静态成员
JS中的公有成员,私有成员和静态成员 a.实现类的公有成员 b.实现类的私有成员 c.实现类的静态成员 a.实现类的公有成员 之前定义的任何类型成员都属于公有成员的范畴,该类的任何实例都对外公开这些属 ...
- C++类中的常成员和静态成员
常变量.常对象.常引用.指向常对象或常变量的指针等在定义时都使用了const关键字,这是C++语言引入的一种数据保护机制,称为const数据保护机制.例如通过const关键字主动地将被调函数形参进行限 ...
- 精读JavaScript模式(七),命名空间模式,私有成员与静态成员
一.前言 惰性十足,这篇2月19号就开始写了,拖到了现在,就是不愿意花时间把看过的东西整理一下,其它的任何事都比写博客要有吸引力,我要反省自己. 从这篇开始,是关于JS对象创建模式的探讨,JS语言简单 ...
- C++ 静态数据成员和静态成员函数
一 静态数据成员: 1.静态数据成员的定义. 静态数据成员实际上是类域中的全局变量.所以,静态数据成员的定义(初始化)不应该被放在头文件中,因为这样做会引起重复定义这样的错误.即使加上#ifndef ...
- js 独立命名空间,私有成员和静态成员
独立的命名空间 1可以避免全局变量污染. 全局变量污染不是 说 被全局变量污染,而是说不会污染全局变量. 2实现私有成员. 在js中函数 就可以满足独立的命名空间的两点需求. 如: ...
- C# 多线程程序隐患
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.T ...
- C++基础(静态数据成员和静态成员函数)
[简介] 1.静态数据成员在类中声明,在源文件中定义并初始化: 2.静态成员函数没有this指针,只能访问静态数据成员: 3.调用静态成员函数:(1)对象.(2)直接调用: 4.静态成员函数的地址可用 ...
随机推荐
- MJRefresh的一个注意事项
如果从视图一跳转到视图二之后,在视图二中进行MJRefresh的刷新操作,那么在推出试图二之前要用dealloc函数将MJRefreshHeaderView或者MJRefreshFooterView释 ...
- JavaScript基础--DOM对象(十三):(windows对象:history\location\navigator\screen\event)
DOM编程1.为什么要学习DOM(1) 通过dom编程,我们可以写出各种网页游戏(2)dom编程也是ajax的重要基础2.DOM编程介绍DOM = Document Object Model(文档对象 ...
- UIImagePickerController 获取相片视频
1.UIImagePickerController的静态方法: imagepicker = [[UIImagePickerController alloc]init]; //UIImagePic ...
- 当一个activity中按钮过多时怎么办?
这几天看极客学院的视频,跟视频中的老师学到的一些小技巧~~ .setOnClickListener(this) 通过重写this(我猜的是重写),下面有onClicked() package exam ...
- mysql 日期类型比较
MySQL 日期类型:日期格式.所占存储空间.日期范围 比较. 日期类型 存储空间 日期格式 日期范围 ------------ ------ ...
- 团队开发——冲刺2.d
冲刺阶段二(第四天) 1.昨天做了什么? 把收集的图标进行统一整理,使用相同风格.类型,使界面更加美观. 2.今天准备做什么? 开始写测试计划书. 3.遇到什么困难? 关于昨天遇到的问题:在游戏界面加 ...
- Structs2中iterator的status属性的用法
iterator标签主要是用于迭代输出集合元素,如list set map 数组等,在使用<s:iterator/>标签的时候有三个属性值得我们关注 1. value属性:可选的属性,va ...
- 第三个Sprint冲刺第七天
讨论地点:宿舍 讨论成员:邵家文.李新.朱浩龙.陈俊金 讨论问题:做最后的工作
- CI框架源码分析
这几天,把ci源码又看了一遍,于是有了新的收获.明白了在application目录下core文件夹的作用,就是用来写ci核心文件的扩展的, 而且需要在配置文件中添加类前缀MY_. CI框架整体是但入口 ...
- MATLAB与C/C++混合编程的一些总结
[转载请注明出处]http://www.cnblogs.com/mashiqi 先上总结: 由于C/C++语言的函数输入输出参数的特点,可以将多个参数方便地传入一个函数中,但却不能方便地返回多个参数. ...