有关Java的日期处理的一些杂记

本文介绍了在Java中使用SimpleDateFormat处理日期时可能遇到的线程安全问题,包括可能导致的错误输出和NumberFormatException。建议在多线程环境中避免使用同一个DateFormat实例,推荐使用JODA-Time或ThreadLocal来解决并发问题。同时,文章提到了FastDateFormat和通过同步方法来确保线程安全的其他策略。
摘要由CSDN通过智能技术生成

在企业应用开发中,经常会遇到日期的相关处理,说实话JDK自带的日期方法很难用。就我个人而言我一般都会采用joda-time来替代JDK自身的日期。

SimpleDateFormat

先来说说JDK自带的SimpleDateFormat类吧。SimpleDateFormat 是 Java 中一个非常常用的类用来对日期字符串进行解析和格式化输出,但如果使用不小心会导致非常微妙和难以调试的问题,因为 DateFormat 和 SimpleDateFormat 类不都是线程安全的,在多线程环境下调用 format() 和 parse() 方法应该使用同步代码来避免问题。

下面是你在使用 SimpleDateFormat 应该要小心的几点:

确保不会在多线程状态下使用同一个 DateFormat 或者 SimpleDateFormat 实例
如果多线程情况下需要访问同一个实例,那么请用同步方法
可以使用 JODA 日期时间处理库来避免这些问题
你也可以使用 commons-lang 包中的 FastDateFormat 工具类
另外你也可以使用 ThreadLocal 来处理这个问题
下面我们通过代码来说明上面的问题:

以下的代码为我们展示了如何在一个线程环境里面使用DateFormat把字符串日期转换为日期对象。创建一个实例来获取日期格式会比较高效,因为系统不需要多次获取本地语言和国家。

public class DateFormatTest {
  
  private final DateFormat format =
            new SimpleDateFormat("yyyyMMdd");
  
  public Date convert(String source)
                      throws ParseException{
    Date d = format.parse(source);
    return d;
  }
}

这段代码是非线程安全的。我们可以通过在多个线程中调用它。在以下调用的代码中,我创建了一个有两个线程的线程池,并提交了5个日期转换任务,之后查看运行结果:

final DateFormatTest t =new DateFormatTest();
Callable<Date> task =new Callable<Date>(){
    public Date call()throws Exception {
        return t.convert("20100811");
    }//需要获取资料的朋友请加Q君样:290194256*
};
  
//让我们尝试2个线程的情况
ExecutorService exec = Executors.newFixedThreadPool(2);
List<Future<Date>> results =
             new ArrayList<Future<Date>>();
  
//实现5次日期转换
for(int i =0; i <5; i++){
    results.add(exec.submit(task));
}
exec.shutdown();
  
//查看结果
for(Future<Date> result : results){
    System.out.println(result.get());
}

代码的运行结果并非如我们所愿 - 有时候,它输出正确的日期,有时候会输出错误的(例如.Sat Jul 31 00:00:00 BST 2012),有些时候甚至会抛出NumberFormatException!

如何并发使用DateFormat类

我们可以有多种方法在线程安全的情况下使用DateFormat类。

1. 同步

最简单的方法就是在做日期转换之前,为DateFormat对象加锁。这种方法使得一次只能让一个线程访问DateFormat对象,而其他线程只能等待。

public Date convert(String source)
                    throws ParseException{
  synchronized(format) {
    Date d = format.parse(source);
    return d;
  }//需要获取资料的朋友请加Q君样:290194256*
}
  

2. 使用ThreadLocal

另外一个方法就是使用ThreadLocal变量去容纳DateFormat对象,也就是说每个线程都有一个属于自己的副本,并无需等待其他线程去释放它。这种方法会比使用同步块更高效。

public class DateFormatTest {
  
  private static final ThreadLocal<DateFormat> df
                 = new ThreadLocal<DateFormat>(){
    @Override
    protected DateFormat initialValue() {
        return new SimpleDateFormat("yyyyMMdd");
    }//需要获取资料的朋友请加Q君样:290194256*
  };
  
  public Date convert(String source)
                     throws ParseException{
    Date d = df.get().parse(source);
    return d;
  }
}

3. Joda-Time

Joda-Time 是一个很棒的开源的 JDK 的日期和日历 API 的替代品,其 DateTimeFormat 是线程安全而且不变的。

import org.joda.time.DateTime;
import org.joda.time.format.DateTimeFormat;
import org.joda.time.format.DateTimeFormatter;
import java.util.Date;
  
public class DateFormatTest {
  
  private final DateTimeFormatter fmt =
       DateTimeFormat.forPattern("yyyyMMdd");
  
  public Date convert(String source){
    DateTime d = fmt.parseDateTime(source);
    returnd.toDate();
  }
}

在这里插入图片描述
 在这里插入图片描述
最新2020整理收集的一些高频面试题(都整理成文档),有很多干货,包含mysql,netty,spring,线程,spring cloud、jvm、源码、算法等详细讲解,也有详细的学习规划图,面试题整理等,需要获取这些内容的朋友请加Q君样:290194256*

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值