DateTimeFormatter 与SimpleDateFormat 的区别

SimpleDateFormat 是线程不安全的类,一般不要定义为static变量,如果定义为static,必须加锁,或者使用DateUtils工具类。这是阿里java开发规范提到 的强制要求。为何如此呢?

  • SimpleDateFormat 是非线程安全类,在使用过程中可能会造成安全问题。多线程调用parse方法剖出的异常:

  • DateTimeFormatter 与SimpleDateFormat 的区别_第1张图片

  • DateTimeFormatter  显示说明此类是不可变的且是线程安全的类。

  • DateTimeFormatter 与SimpleDateFormat 的区别_第2张图片

  • SimpleDateFormat  的不安全使用:如果多个线程使用同一个SimpleDateFormat 的对象进行日期和字符串的转化,则会有异常抛出。

  • 所以如果使用SimpleDateFormat 在多线程下使用,则应该使用threadLocal,为每一个线程构造一个对象。这样就避免了多线程的竞争条件,从而可以安全使用。

private static final ThreadLocal df = new ThreadLocal() {
@Override
protected DateFormat initialValue() {
return new SimpleDateFormat("yyyy-MM-dd");
}
};
  • DateTimeFormatter  的parse为何是线程安全的呢?由于它调用的parseResolved0 方法如下,可以看出这个方法的入参都是final 修饰的,不可变变量是线程安全的。
  • DateTimeFormatter 与SimpleDateFormat 的区别_第3张图片
  • DateTimeFormatter   的使用,通过如下测试也可看到结果。
  •    DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");
    
            SimpleDateFormat sdf=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
    
            for (int i = 0; i < 1000; i++) {
                new Thread(new Runnable() {
                    @Override
                    public void run() {
                        TemporalAccessor parse = dateTimeFormatter.parse("2019-05-19 14:22:52");
                        LocalDateTime from = LocalDateTime.from(parse);
                        Instant second = from.toInstant(ZoneOffset.ofTotalSeconds(0));
                        System.out.println(second);
                    }
                }).start();
            }

     

你可能感兴趣的:(java)