`
lydawen
  • 浏览: 464612 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

SimpleDateFormat使用不当导致时间错乱

    博客分类:
  • java
 
阅读更多

近期一个老项目发现导入数据(节目单),存在时间错乱的问题。节目单来自文件,文件格式大概:

中央五套.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 这样错误的时间。

 

 

 

分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics