近期一个老项目发现导入数据(节目单),存在时间错乱的问题。节目单来自文件,文件格式大概:
中央五套.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 这样错误的时间。