`
lydawen
  • 浏览: 471758 次
  • 性别: 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 这样错误的时间。

 

 

 

分享到:
评论

相关推荐

    SimpleDateFormat使用详解

    使用 SimpleDateFormat 可以实现日期和时间的格式化、解析和标准化。例如,使用 SimpleDateFormat 可以将 Date 对象格式化为字符串,或者将字符串解析为 Date 对象。 下面是一个简单的示例代码: ```java import ...

    java SimpleDateFormat 显示于系统时间不符

    SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); date.setTime(time); System.out.println(sdf.format(date)); 发现时间于想要的时间不符,请运行Time.reg文件

    深入理解Java:SimpleDateFormat安全的时间格式化

    但是,如果使用不小心会导致非常微妙和难以调试的问题,因为DateFormat和SimpleDateFormat类不是线程安全的。在多线程环境下调用format()和parse()方法应该使用同步代码来避免问题。 知识点1: SimpleDateFormat的非...

    java 使用SimpleDateFormat类获取系统的当前时间

    下面我们将详细探讨如何使用 `SimpleDateFormat` 类获取系统的当前时间。 首先,我们需要了解 `java.util.Date` 类,它是Java中表示日期和时间的基础类。`Date` 类有一个方法 `now()`,可以用来获取当前系统的日期...

    Java 实例 - 格式化时间SimpleDateFormat使用源代码-详细教程.zip

    使用`SimpleDateFormat`时,你需要定义一个模式字符串,它描述了日期和时间的格式。例如,"yyyy-MM-dd HH:mm:ss"将日期格式化为"年-月-日 时:分:秒"。模式字符如'y'代表年,'M'代表月份,'d'代表日期,'H'代表24...

    高并发之-SimpleDateFormat类的线程安全问题和解决方案.docx

    SimpleDateFormat类的线程安全问题是由于它使用了缓存机制导致的。在高并发的情况下,SimpleDateFormat类会出现线程安全问题。为了解决这个问题,可以使用ThreadLocal、同步机制、线程安全的日期时间转化类或池化...

    由浅入深解析 SimpleDateFormat

    例如,在导出数据库数据到 excel 文件时,我们可以使用 SimpleDateFormat 来格式化日期和时间。 简单DateFormat 的实例 在上面的例子中,我们创建了一个 DateUtil 类,用于格式化日期和时间。在这个类中,我们使用...

    Java SimpleDateFormat中英文时间格式化转换详解

    本文主要为大家详细介绍了Java SimpleDateFormat中英文时间格式化转换的知识点,包括日期和时间模式的设定、SimpleDateFormat的使用、日期和时间格式的转换等。 日期和时间模式 在Java中,日期和时间模式是通过...

    simpleDateFormat是线程不安全的

    在Java编程语言中,`SimpleDateFormat`类是一个广泛使用的日期时间格式化工具,但它的线程安全性是一个常常被开发者忽视的问题。标题指出的"simpleDateFormat是线程不安全的",意味着在多线程环境下,如果多个线程...

    JAVA使用SimpleDateFormat类表示时间代码实例

    JAVA 使用 SimpleDateFormat 类表示时间代码实例 JAVA 中的日期和时间处理是非常重要的,特别是在程序开发中,经常需要处理日期和时间的相关数据。在 JAVA 中,我们可以使用 java.util 包中的 Date 类来获取当前...

    有关SimpleDateFormat的常用方法说明

    - 在使用`SimpleDateFormat`时,需要注意日期和时间的格式是否符合预期,特别是在国际化应用中,不同的地区可能会有不同的日期和时间格式习惯。 - `SimpleDateFormat`是线程不安全的,如果在多线程环境中使用,需要...

    JavaScript实现的SimpleDateFormat

    在JavaScript中,没有内置的`SimpleDateFormat`类,它是一个Java中的类,用于日期和时间的格式化。然而,由于JavaScript在处理日期时的功能相对有限,开发者常常需要自定义函数或者使用第三方库来实现类似的功能。这...

    java SimpleDateFormat &Calendar

    在Java编程语言中,`SimpleDateFormat`和`Calendar`是两个重要的日期和时间处理类,它们在处理日期格式化、解析以及日期计算方面扮演着重要角色。本文将深入探讨这两个类的功能、用法以及它们之间的关系。 `...

    Java中的SimpleDateFormat使用详解

    `SimpleDateFormat`的基本使用方法是创建一个新的实例,并通过传递一个字符串模式来设定日期和时间的格式。这个模式字符串由一系列特定的字母和符号组成,每个代表日期或时间的一部分。例如: - `G`: 年代标识符,...

    SimpleDateFormat格式化日期

    日期操作。。。基础的SimpleDateFormat格式化日期!!操作!》初级学习代码

    java日期格式化SimpleDateFormat的使用详解

    以下是一个使用 SimpleDateFormat 将日期和时间格式化为字符串的示例代码: ```java private static void formatDataTest() { StringBuilder sb = new StringBuilder(); sb.append("yyyy年MM月dd日 HH:mm:ss") ....

    Java中SimpleDateFormat的使用方法

    主要为大家详细介绍了Java中SimpleDateFormat的使用方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

    创建SimpleDateFormat对象,确定日期被格式化的格式.txt

    1.创建SimpleDateFormat对象,确定日期被格式化的格式 2.使用循环,在循环中调用Thread的sleep方法,让线程休眠1s后打印当前时间的字符串

    ThreadLocal:如何优雅的解决SimpleDateFormat多线程安全问题

    目录SimpleDateFormat诡异bug复现SimpleDateFormat诡异bug字符串日期转Date日期(parse)Date日期转String类型(format)SimpleDateFormat出现bug的原因如何解决SimpleDateFormat多线程安全问题局部变量使用...

Global site tag (gtag.js) - Google Analytics