Java多线程编程中使用DateFormat类
DateFormat类是一个非线程安全的类。javadocs文档里面提到"Dateformats是不能同步的。我们建议为每个线程创建独立的日期格式。如果多个线程同时访问一个日期格式,这需要在外部加上同步代码块。"
以下的代码为我们展示了如何在一个线程环境里面使用DateFormat把字符串日期转换为日期对象。创建一个实例来获取日期格式会比较高效,因为系统不需要多次获取本地语言和国家。
publicclassDateFormatTest{
privatefinalDateFormatformat=
newSimpleDateFormat("yyyyMMdd");
publicDateconvert(Stringsource)
throwsParseException{
Dated=format.parse(source);
returnd;
}
}
这段代码是非线程安全的。我们可以通过在多个线程中调用它。在以下调用的代码中,我创建了一个有两个线程的线程池,并提交了5个日期转换任务,之后查看运行结果:
finalDateFormatTestt=newDateFormatTest();
Callable<Date>task=newCallable<Date>(){
publicDatecall()throwsException{
returnt.convert("20100811");
}
};
//让我们尝试2个线程的情况
ExecutorServiceexec=Executors.newFixedThreadPool(2);
List<Future<Date>>results=
newArrayList<Future<Date>>();
//实现5次日期转换
for(inti=0;i<5;i++){
results.add(exec.submit(task));
}
exec.shutdown();
//查看结果
for(Future<Date>result:results){
System.out.println(result.get());
}
代码的运行结果并非如我们所愿-有时候,它输出正确的日期,有时候会输出错误的(例如.SatJul3100:00:00BST2012),有些时候甚至会抛出NumberFormatException!
如何并发使用DateFormat类
我们可以有多种方法在线程安全的情况下使用DateFormat类。
1.同步
最简单的方法就是在做日期转换之前,为DateFormat对象加锁。这种方法使得一次只能让一个线程访问DateFormat对象,而其他线程只能等待。
publicDateconvert(Stringsource)
throwsParseException{
synchronized(format){
Dated=format.parse(source);
returnd;
}
}
2.使用ThreadLocal
另外一个方法就是使用ThreadLocal变量去容纳DateFormat对象,也就是说每个线程都有一个属于自己的副本,并无需等待其他线程去释放它。这种方法会比使用同步块更高效。
publicclassDateFormatTest{
privatestaticfinalThreadLocal<DateFormat>df
=newThreadLocal<DateFormat>(){
@Override
protectedDateFormatinitialValue(){
returnnewSimpleDateFormat("yyyyMMdd");
}
};
publicDateconvert(Stringsource)
throwsParseException{
Dated=df.get().parse(source);
returnd;
}
}
3.Joda-Time
Joda-Time是一个很棒的开源的JDK的日期和日历API的替代品,其DateTimeFormat是线程安全而且不变的。
importorg.joda.time.DateTime;
importorg.joda.time.format.DateTimeFormat;
importorg.joda.time.format.DateTimeFormatter;
importjava.util.Date;
publicclassDateFormatTest{
privatefinalDateTimeFormatterfmt=
DateTimeFormat.forPattern("yyyyMMdd");
publicDateconvert(Stringsource){
DateTimed=fmt.parseDateTime(source);
returnd.toDate();
}
}