SimpleDateFormat做成员或者静态成员多线程安全隐患
来源:互联网 发布:猜骰子源码 编辑:程序博客网 时间:2024/05/16 14:23
有时我们在同一个类中都是使用同一种日期格式,又或者为了减少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做成员或者静态成员多线程安全隐患
- SimpleDateFormat做静态成员 成员 多线程并发出现问题
- 自定义类做成员变量
- C++ 类的静态成员、友员函数
- SimpleDateFormat 引起的线程安全隐患问题
- 多线程的安全隐患
- iOS 【多线程安全隐患】
- 多线程的安全隐患
- 多线程的安全隐患
- 用点类做成员的三角形类
- Static 静态成员函数-与多线程
- Static 静态成员函数与多线程
- 基础-函数-静态方法、非静态方法、静态成员、非静态成员与多线程
- 静态员初始化
- 静态方法和静态成员变量和多线程
- SimpleDateFormat多线程问题
- SimpleDateFormat多线程问题
- SimpleDateFormat多线程调用问题
- (转载)SSH错误1:The import javax.servlet.http.HttpServletRequest cannot be resolved
- 【转】对于学习方式的一些思考
- 当有多个设备online时,命令行窗口通过adb连接指定设备方法
- java IO (三) 字节流与缓冲
- 制约程序员"钱途"的两大最关键因素
- SimpleDateFormat做成员或者静态成员多线程安全隐患
- 联合主键二:组件映射方式
- .net 中异步SOCKET发送数据时碰到的内存问题 (二)
- 编译到执行 关于内存分配
- 一个人的Scrum之准备工作
- OpenGL函数
- cmem模块/DVSDK2.0
- Qt for android
- Oracle数据库通过undo保证一致性读和不发生脏读