近期一个老项目发现导入数据(节目单),存在时间错乱的问题。节目单来自文件,文件格式大概:
中央五套.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 可以将 Date 对象格式化为字符串,或者将字符串解析为 Date 对象。 下面是一个简单的示例代码: ```java import ...
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); date.setTime(time); System.out.println(sdf.format(date)); 发现时间于想要的时间不符,请运行Time.reg文件
但是,如果使用不小心会导致非常微妙和难以调试的问题,因为DateFormat和SimpleDateFormat类不是线程安全的。在多线程环境下调用format()和parse()方法应该使用同步代码来避免问题。 知识点1: SimpleDateFormat的非...
下面我们将详细探讨如何使用 `SimpleDateFormat` 类获取系统的当前时间。 首先,我们需要了解 `java.util.Date` 类,它是Java中表示日期和时间的基础类。`Date` 类有一个方法 `now()`,可以用来获取当前系统的日期...
使用`SimpleDateFormat`时,你需要定义一个模式字符串,它描述了日期和时间的格式。例如,"yyyy-MM-dd HH:mm:ss"将日期格式化为"年-月-日 时:分:秒"。模式字符如'y'代表年,'M'代表月份,'d'代表日期,'H'代表24...
SimpleDateFormat类的线程安全问题是由于它使用了缓存机制导致的。在高并发的情况下,SimpleDateFormat类会出现线程安全问题。为了解决这个问题,可以使用ThreadLocal、同步机制、线程安全的日期时间转化类或池化...
例如,在导出数据库数据到 excel 文件时,我们可以使用 SimpleDateFormat 来格式化日期和时间。 简单DateFormat 的实例 在上面的例子中,我们创建了一个 DateUtil 类,用于格式化日期和时间。在这个类中,我们使用...
本文主要为大家详细介绍了Java SimpleDateFormat中英文时间格式化转换的知识点,包括日期和时间模式的设定、SimpleDateFormat的使用、日期和时间格式的转换等。 日期和时间模式 在Java中,日期和时间模式是通过...
在Java编程语言中,`SimpleDateFormat`类是一个广泛使用的日期时间格式化工具,但它的线程安全性是一个常常被开发者忽视的问题。标题指出的"simpleDateFormat是线程不安全的",意味着在多线程环境下,如果多个线程...
JAVA 使用 SimpleDateFormat 类表示时间代码实例 JAVA 中的日期和时间处理是非常重要的,特别是在程序开发中,经常需要处理日期和时间的相关数据。在 JAVA 中,我们可以使用 java.util 包中的 Date 类来获取当前...
- 在使用`SimpleDateFormat`时,需要注意日期和时间的格式是否符合预期,特别是在国际化应用中,不同的地区可能会有不同的日期和时间格式习惯。 - `SimpleDateFormat`是线程不安全的,如果在多线程环境中使用,需要...
在JavaScript中,没有内置的`SimpleDateFormat`类,它是一个Java中的类,用于日期和时间的格式化。然而,由于JavaScript在处理日期时的功能相对有限,开发者常常需要自定义函数或者使用第三方库来实现类似的功能。这...
在Java编程语言中,`SimpleDateFormat`和`Calendar`是两个重要的日期和时间处理类,它们在处理日期格式化、解析以及日期计算方面扮演着重要角色。本文将深入探讨这两个类的功能、用法以及它们之间的关系。 `...
`SimpleDateFormat`的基本使用方法是创建一个新的实例,并通过传递一个字符串模式来设定日期和时间的格式。这个模式字符串由一系列特定的字母和符号组成,每个代表日期或时间的一部分。例如: - `G`: 年代标识符,...
日期操作。。。基础的SimpleDateFormat格式化日期!!操作!》初级学习代码
以下是一个使用 SimpleDateFormat 将日期和时间格式化为字符串的示例代码: ```java private static void formatDataTest() { StringBuilder sb = new StringBuilder(); sb.append("yyyy年MM月dd日 HH:mm:ss") ....
主要为大家详细介绍了Java中SimpleDateFormat的使用方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
1.创建SimpleDateFormat对象,确定日期被格式化的格式 2.使用循环,在循环中调用Thread的sleep方法,让线程休眠1s后打印当前时间的字符串
目录SimpleDateFormat诡异bug复现SimpleDateFormat诡异bug字符串日期转Date日期(parse)Date日期转String类型(format)SimpleDateFormat出现bug的原因如何解决SimpleDateFormat多线程安全问题局部变量使用...