近期一个老项目发现导入数据(节目单),存在时间错乱的问题。节目单来自文件,文件格式大概:
中央五套.txt
中央五套 2011-01-01
18:30:00 CBA
19:00:00 新闻联播
……
中央六套.txt
中央六套 2011-01-01
10:00:00 大漠英雄
12:00:00 午间新闻
……
电视有很多频道,偶尔存在1-3人同时导入不同频道节目的情况,有时会出来18:30:00 大漠英雄,即时间已经串掉了,甚至有时会出来2012-02-31 这种变态的情况,老系统遗留已久今天头也下决心查,结果发现有如下代码
final static SimpleDateFormat sdf=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
而将字符串转换为Date 的代码都使用此方式
Date date=sdf.parse("2011-01-01 18:30:00");
SimpleDateFormat的parse实际是继承自DateFormat类的方法,代码如下:
public Date parse(String source) throws ParseException
{
ParsePosition pos = new ParsePosition(0);
Date result = parse(source, pos);
if (pos.index == 0)
throw new ParseException("Unparseable date: \"" + source + "\"" ,
pos.errorIndex);
return result;
}
而上面的parse(source,pos)则是使用实现类的方法,这里即SimpleDateFormat.parse(source,pos)方法,它的代码就不贴了,但可以明显的发现SimpleDateFormat继承自DateFormat的Calendar calendar,在private int subParse(String text, int start, int patternCharIndex, int count,
boolean obeyCount, boolean[] ambiguousYear,
ParsePosition origPos) 方法中,会解析传入的source,并依次将解析出的yyyy,MM,dd,HH,mm,ss设置到calendar中,由于calendar在这是实例变量,所以在多线程的情况下就会造成数据串的情况,接下来写个测试类来模拟:
public class TestSDF {
final static SimpleDateFormat sdf=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
public String format(Date d){
return sdf.format(d);
}
public static void main(String[] args) {
ExecutorService es=Executors.newFixedThreadPool(50);
for(int i=0;i<50;i++){
PrintTimeNew pt=new PrintTimeNew();
pt.threadName=i;
es.execute(pt);
}
es.shutdown();
}
}
class PrintTimeNew extends Thread{//时间转字符线程
public int threadName;
TestSDF ts=new TestSDF();
SimpleDateFormat sdf=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
@Override
public void run() {
super.run();
Date d=null;
String strD=null;
try{
if(threadName%2==0){
strD="2012-02-02 12:12:12";
}else{
strD="2012-03-30 12:12:12";
}
d=sdf.parse(strD);
}catch (Exception e) {
}
for(int i=0;i<100;i++){
String sd=ts.format(d);
try {
Thread.sleep(1);
} catch (Exception e) {
}
if(!sd.equals("2012-02-02 12:12:12") && !sd.equals("2012-03-30 12:12:12")){
System.out.println(threadName+": 原:"+strD+" 新:"+sd);
}
}
}
}
运行代码后,基本都能重现几十次并发问题,如下(部分)运行结果:
2: 原:2012-02-02 12:12:12 新:2012-02-30 12:12:12
1: 原:2012-03-30 12:12:12 新:2012-03-02 12:12:12
20: 原:2012-02-02 12:12:12 新:2012-02-30 12:12:12
17: 原:2012-03-30 12:12:12 新:2012-03-02 12:12:12
48: 原:2012-02-02 12:12:12 新:2012-02-30 12:12:12
42: 原:2012-02-02 12:12:12 新:2012-02-30 12:12:12
40: 原:2012-02-02 12:12:12 新:2012-02-30 12:12:12
34: 原:2012-02-02 12:12:12 新:2012-02-30 12:12:12
如上,就出现了2012-02-30 这样错误的时间。
分享到:
相关推荐
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); date.setTime(time); System.out.println(sdf.format(date)); 发现时间于想要的时间不符,请运行Time.reg文件
java 使用SimpleDateFormat类获取系统的当前时间 java 使用SimpleDateFormat类获取系统的当前时间
SimpleDateFormat使用详解。非常实用!!!!
深入理解Java:SimpleDateFormat安全的时间格式化
Java 实例 - 格式化时间SimpleDateFormat使用源代码-详细教程.zip
有关SimpleDateFormat的常用方法说明
JavaScript实现的java.text.SimpleDateFormat。希望多多交流。
主要介绍了JAVA使用SimpleDateFormat类表示时间代码实例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
主要为大家详细介绍了Java SimpleDateFormat中英文时间格式化转换,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
日期操作。。。基础的SimpleDateFormat格式化日期!!操作!》初级学习代码
由浅入深解析 SimpleDateFormat 由浅入深解析 SimpleDateFormat
SimpleDateFormat 是一个以国别敏感的方式格式化和分析数据的具体类。这篇文章主要介绍了Java中的SimpleDateFormat使用详解,需要的朋友可以参考下
主要为大家详细介绍了Java中SimpleDateFormat的使用方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
目录SimpleDateFormat诡异bug复现SimpleDateFormat诡异bug字符串日期转Date日期(parse)Date日期转String类型(format)SimpleDateFormat出现bug的原因如何解决SimpleDateFormat多线程安全问题局部变量使用...
1.创建SimpleDateFormat对象,确定日期被格式化的格式 2.使用循环,在循环中调用Thread的sleep方法,让线程休眠1s后打印当前时间的字符串
主要介绍了java SimpleDateFormat使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
高并发之-SimpleDateFormat类的线程安全问题和解决方案.docx
NULL 博文链接:https://qinchaoyong.iteye.com/blog/543109
计算机后端-Java-Java核心基础-第22章 常用类 10. SimpleDateFormat的使用.avi