SimpleDateFormat线程安全问题深入解析_simpledateformat为什么会有线程安全-程序员宅基地

技术标签: Java  Source Code  多线程  线程安全  SimpleDateFormat  

背景

众所周知,Java中的SimpleDateFormat不是线程安全的,在多线程下会出现意想不到的问题。本文将解析SimpleDateFormat线程不安全的具体原因,从而加深对线程安全的理解。

例子

简单的测试代码,当多个线程同时调用parse方法的时候会出问题:

public class SimpleDateFormatTest {
    
    private static SimpleDateFormat format = new SimpleDateFormat("yyyy/MM/dd HH:mm:ss");

    public static void main(String[] args) {
    
        for (int i = 0; i < 20; i++) {
    
            new Thread(() -> {
    
                try {
    
                    System.out.println(format.parse("2019/11/11 11:11:11"));
                } catch (ParseException e) {
    
                    e.printStackTrace();
                }
            }).start();
        }
    }
}

部分输出如下:

Mon Nov 11 11:11:11 GMT 2019
Thu Jan 01 00:00:00 GMT 1970
java.lang.NumberFormatException: For input string: ""
	at java.lang.NumberFormatException.forInputString(NumberFormatException.java:65)
	at java.lang.Long.parseLong(Long.java:601)
	at java.lang.Long.parseLong(Long.java:631)
	at java.text.DigitList.getLong(DigitList.java:195)
	at java.text.DecimalFormat.parse(DecimalFormat.java:2051)
	at java.text.SimpleDateFormat.subParse(SimpleDateFormat.java:2162)
	at java.text.SimpleDateFormat.parse(SimpleDateFormat.java:1514)
	at java.text.DateFormat.parse(DateFormat.java:364)
	at package1.SimpleDateFormatTest.lambda$0(SimpleDateFormatTest.java:17)
	at package1.SimpleDateFormatTest
	at java.lang.Thread.run(Thread.java:745)
java.lang.NumberFormatException: empty String
	at sun.misc.FloatingDecimal.readJavaFormatString(FloatingDecimal.java:1842)
	at sun.misc.FloatingDecimal.parseDouble(FloatingDecimal.java:110)
	at java.lang.Double.parseDouble(Double.java:538)
	at java.text.DigitList.getDouble(DigitList.java:169)
	at java.text.DecimalFormat.parse(DecimalFormat.java:2056)
	at java.text.SimpleDateFormat.subParse(SimpleDateFormat.java:2162)
	at java.text.SimpleDateFormat.parse(SimpleDateFormat.java:1514)
	at java.text.DateFormat.parse(DateFormat.java:364)
	at package1.SimpleDateFormatTest.lambda$0(SimpleDateFormatTest.java:17)
	at package1.SimpleDateFormatTest
	at java.lang.Thread.run(Thread.java:745)

不出意外,每次跑都会报错,偶尔还会出现输出初始时间Thu Jan 01 00:00:00 GMT 1970以及其他莫名其妙的时间。好的,记住这两个错误,下面我们仔细分析。

分析

SimpleDateFormat继承自DateFormat这个抽象类,UML图如下:
SimpleDateFormat UML

DateFormat中有两个全局变量需要注意

public abstract class DateFormat extends Format {
    

    //日历变量,作为DateFormat的辅助
    protected Calendar calendar;

    //用来Format数字,默认为DecimalFormat
    protected NumberFormat numberFormat;
}

public class DecimalFormat extends NumberFormat {
    
    //DecimalFormat中的全局变量,用来存放转化好的数据
    //digitList用科学技计数表示,如2019表示成0.2019x10^4
    private transient DigitList digitList = new DigitList();
}

这两个变量的初始化在SimpleDateFormat的构造方法里初始化。
看了类结构,我们仔细分析一下DateFormatparse方法,直接上代码(省略掉了一些无关紧要的代码):

public Date parse(String text, ParsePosition pos)
{
    
    ......
    //注意这个变量calb,日期的转化是通过CalendarBuilder这个类来完成的
    CalendarBuilder calb = new CalendarBuilder();

    //按照DateFormat的pattern逐个循环(年月日时分秒...)
    for (int i = 0; i < compiledPattern.length; ) {
    
        ......
        //最终调用subParse方法给calb赋值
        start = subParse(text, start, tag, count, obeyCount, ambiguousYear, pos, useFollowingMinusSignAsDelimiter, calb);
    }
    Date parsedDate;
    try {
    
        //调用CalendarBuilder的establish方法,把值传递给变量calendar
        //通过calendar来获取最终返回的日期
        //注意,这里calendar是个全局变量
        parsedDate = calb.establish(calendar).getTime();
    }
    ......

    return parsedDate;
}

主要分为如下几个步骤:

  1. 定义一个CalendarBuilder对象calb,用来临时保存parse结果。
  2. 根据DateFormat定义的Pattern,for循环调用subParse方法,将目标字符串逐个(年月日时分秒…)转化,并存储在calb变量里。
  3. 调用calb.establish(calendar)方法,把暂存在calb里的数据设置到全局变量calendar里。
  4. 现在calendar里已经包含转换过的日期数据,最后调用**Calendar.getTime()**方法返回日期。
问题之一

下面看一下subParse方法里面做了什么,实现上有什么问题。先看代码(省略掉了一些无关紧要的代码):

public class SimpleDateFormat extends DateFormat {
    
    private int subParse(String text, int start, int patternCharIndex, int count,
                    boolean obeyCount, boolean[] ambiguousYear,
                    ParsePosition origPos,
                    boolean useFollowingMinusSignAsDelimiter, CalendarBuilder calb) {
    
        //一些变量初始化
        ......

        //内部调用numberFormat的parse方法,转化数字
        //这里的numberFormat就是上面分析过的那个全局变量,默认实例是DecimalFormat
        //text是代转字符串"2019/11/11 11:11:11", pos是位置,如2019会被转化为0.2019x10^4
        number = numberFormat.parse(text, pos);
        if (number != null) {
    
            //转化成int值,如0.2019x10^4会转化成2019
            value = number.intValue();
        }
        int index;
        switch (patternCharIndex) {
    
        case PATTERN_YEAR:      // 'y'
            //有年,月,日等等各种case,这里只拿PATTERN_YEAR(年)这种情况举例子
            //将numberFormat parse出来的值set到calb里面去
            calb.set(field, value);
            return pos.index;
        }

        ......

        // 转义失败
        origPos.errorIndex = pos.index;
        return -1;
    }
}

//numberFormat.parse(text, pos)方法实现
public class DecimalFormat extends NumberFormat {
    

    public Number parse(String text, ParsePosition pos) {
    
        //内部调用subparse方法,将text的内容set到digitList上
        if (!subparse(text, pos, positivePrefix, negativePrefix, digitList, false, status)) {
    
            return null;
        }
        ......

        //将digitList转变为目标格式
        if (digitList.fitsIntoLong(status[STATUS_POSITIVE], isParseIntegerOnly())) {
    
            //parse为Long型
            longResult = digitList.getLong();
        } else {
    
            //parse为double型
            doubleResult = digitList.getDouble();
        }
        .....

        return gotDouble ? (Number)new Double(doubleResult) : (Number)new Long(longResult);
    }

    private final boolean subparse(String text, ParsePosition parsePosition,
                String positivePrefix, String negativePrefix,
                DigitList digits, boolean isExponent,
                boolean status[]) {
    
        //一些判断及变量初始化准备
        ......

        //digitList在这个方法里面叫digits,先对digits先清零处理。
        //decimalAt指小数点位置,如0.2019x10^4中decimalAt就是4
        //count指数字位数,如0.2019x10^4中count就是4
        digits.decimalAt = digits.count = 0;

        backup = -1;
        for (; position < text.length(); ++position) {
    
            //循环内部对digits一顿猛如虎的赋值操作,设置科学计数法各个部分的变量
            //注意这个digits是一个全局变量
            ......
        }

        //还要对digits继续操作
        if (!sawDecimal) {
    
            digits.decimalAt = digitCount; // Not digits.count!
        }
        digits.decimalAt += exponent;

        ......
        return true;
    }
}

看到这里,有点并发编程经验的同学估计就能看出问题了。在subparse这个方法里面不加保护,当多个线程同时对全局变量digits(digitList)进行操作时,这个变量很可能是个无效的值。比如线程A把值设置了一半,另一个线程B把值又清零初始化了。于是线程A在后面调用digitList.getDoubledigitList.getLong方法的时候要么得到意料之外的值,要么直接报错NumberFormatException

问题之二

那么后面的步骤有没有问题呢?继续往下看。
前面说到,方法会先把parse好的值放到CalendarBuilder型的临时变量calb里面,然后调用establish方法,将calb中缓存的值设置到SimpleDateFormatcalendar变量中,下面看看establish方法:

class CalendarBuilder {
    
    Calendar establish(Calendar cal) {
    
        ......
        //这个cal是SimpleDateFormat中的成员变量calendar
        //先将cal中的数据清除初始化,跟上面digitList一样的套路
        cal.clear();
        
        for (int stamp = MINIMUM_USER_STAMP; stamp < nextStamp; stamp++) {
    
            for (int index = 0; index <= maxFieldIndex; index++) {
    
                if (field[index] == stamp) {
    
                    //前面CalendarBuild暂存的值都放在field数组里,
                    //这里将数组中的值逐个赋给cal
                    cal.set(index, field[MAX_FIELD + index]);
                    break;
                }
            }
        }

        if (weekDate) {
    
            //设置cal的weekdate field
            cal.setWeekDate(field[MAX_FIELD + WEEK_YEAR], weekOfYear, dayOfWeek);
        }
        return cal;
    }
}

还是同样的问题,由于calendar(cal)是个全局变量,当多个线程同时调用establish方法的时候,会有线程安全问题。举个简单的例子,线程A原先赋值好了"2019/11/11 11:11:11",结果线程B调用了cal.clear方法将数据又给清掉了,于是线程A回到了解放前,输出了日期"1970/01/01 00:00:00"。

解决办法

对于线程安全的解决办法,给方法加同步synchronize是最简单的,相当于线程只能一个一个地访问parse方法:

    synchronize (this) {
    
        System.out.println(format.parse("2019/11/11 11:11:11"));
    }

当然更common的使用姿势是配合ThreadLocal使用,相当于给每个线程都定义了一个format变量,线程间互不影响:

    private ThreadLocal<SimpleDateFormat> format = new ThreadLocal<SimpleDateFormat>(){
      
        @Override  
        protected SimpleDateFormat initialValue() {
      
            return new SimpleDateFormat("yyyy/MM/dd HH:mm:ss");  
        }  
    };

    System.out.println(format.get().parse("2019/11/11 11:11:11"));

不过最推荐的还是,不要用SimpleDateFormat,而是用Java8新引入的类LocalDateTime或者DateTimeFormatter,不仅线程安全,而且效率更高。

总结

本文从代码层面分析了SimpleDateFormat线程不安全的原因。subparseestablish两个方法都可能导致问题,前者还会抛出Exception
总结下来,问题都是出在全局变量上。所以当我们定义全局变量的时候一定要谨慎,注意变量是不是线程安全。

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/u010793917/article/details/103449207

智能推荐

python3爬虫学习之实战无限下载百度图片_"re_dbs='\"thumburl\":\"(.*?)"-程序员宅基地

文章浏览阅读3k次。由于做项目需要大量图片,应我同学的要求,帮他爬取大量百度图片当做练习,这里写出来记录一下我们首先分析一下百度图片的页面和获取规则随便输入一个关键字,我们下滑图片页面,同时检查,就可以发现,它是通过Ajax请求一直获取图片之前正好也做了爬取微博的例子,想来是差不多,于是我们分析一下Ajax请求链接我们看到请求URL,多分析几条就能发现,它几乎是不变的,而变化的几条分别是,pn..._"re_dbs='\"thumburl\":\"(.*?)"

windows上如何安装jq_windows jq-程序员宅基地

文章浏览阅读3.6k次。jq介绍- jq是一款非常强大的用来将复杂的数据解析成json格式的工具。具体作用及用法官网有详细介绍。- 官网地址:https://stedolan.github.io/jq/Windows上安装jq- 如果要在Windows上顺利的使用jq,需要先安装chocolatey。- 以管理员身份待开cmd窗口,输入@"%SystemRoot%\System32\WindowsPowerShell\v1.0\powershell.exe" -NoProfile -InputFormat None _windows jq

钉钉官方接口调用过程_钉钉企业内部应用回调怎么弄-程序员宅基地

文章浏览阅读6.5k次,点赞5次,收藏24次。文章目录钉钉官方接口调用过程获取企业内部应用的访问凭证accessToken根据员工的企业UserId获取钉钉官方的unionId唯一性idUserId的获取方法appKey和appSecret的获取方法易错点钉钉官方接口调用过程获取企业内部应用的访问凭证accessToken首先需要引入依赖,如下图:钉钉开发者调用钉钉官方的接口的时候,一定要先要进行认证,什么意思呢?就是必须要告诉钉钉我是一个企业开发者,我有权限调用你的接口,那具体怎么认证呢?企业开发者通常可以在钉钉里面创建应用,我们需要把我们_钉钉企业内部应用回调怎么弄

Ubuntu 8.04嵌入式交叉编译环境arm-linux-gcc搭建过程图解-程序员宅基地

文章浏览阅读136次。Linux版本:Ubuntu8.04内核版本:Linux 2.6.24交叉编译器版本:arm-linux-gcc-3.4.1交叉编译器下载链接:https://share.weiyun.com/5oxlS6X(密码:36R7)前言1、搭建交叉编译环境安装、配置交叉编译工具链。在该环境下编译出嵌入式Linux系统所需的操作系统、应用程序等,然后再上传到目标机上。2、交..._arm-linux-gcc 8

vm虚拟机镜像转换成KVM虚拟机_vm exsi的虚拟机转换成kvm-程序员宅基地

文章浏览阅读1.2k次。众所周知,VMware workstations是一个收费的虚拟机软件,虽然对于个人使用,网上有许多盗版的license可以使用,但是对于一个大的公司来说,在商业环境下,一定不能使用盗版的license,而KVM虚拟机软件是一个免费的软件,在商业应用环境下,完全可以作为一个不错的可以替代VMware workstations的解决方案。但是对于很多已经运行在了VMware workstation的虚拟机镜像来说,要是想避免收费,可以把VMware workstation的虚拟机vmdk的格式转换成qcow_vm exsi的虚拟机转换成kvm

C#接口与继承的区别_c#继承和接口的区别-程序员宅基地

文章浏览阅读3k次。类定义新的数据类型以及这些新的数据类型进行相互操作的方法定义方式: class Cat { } class Cat:object { }C#中所有的类都是默认由object类派生来的,显示指定或者省略效果是一样的,所以上面的两个例子是完全相同的。C#中类包括:抽象类、密封类、非抽象类abstract:表示修饰的类不完整,也就是抽象类,_c#继承和接口的区别

随便推点

Qt调用FFmpeg命令录屏_linux qt 录屏 github-程序员宅基地

文章浏览阅读3.1k次,点赞3次,收藏28次。//列出音视频设备ffmpeg -list_devices true -f dshow -i dummy//录屏命令ffmpeg -f gdigrab -i desktop -f dshow -i audio="麦克风 (High Definition Audio 设备)" -pix_fmt yuv420p -vcodec libx264 -acodec aac -s 1440x900 ..._linux qt 录屏 github

将java swing写的用程序打包exe可执行程序_java swing能生成exe吗-程序员宅基地

文章浏览阅读1.5k次。将java swing写的用程序打包exe可执行程序原文地址:https://www.cnblogs.com/ssh2/p/3463199.html下面的文章是介绍如何将自己编写的swing应用程序打包为.exe的可执行程序,进而编译为.exe的安装程序。相信有这样经历的朋友肯定可以理解我为什么还要将已经编译好的.exe程序更进一步的编译为安装程序,因为一般而言本机打包的是基本不会出现什么..._java swing能生成exe吗

C++之指针探究(十九):typedef和const指针_typedef int* pointer; pointer()-程序员宅基地

文章浏览阅读185次。typedef和const指针 typedef int* pointer; const pointer p1; 相当于是int* const p1; typedef int* pointer; pointer const p1; 相当于是int* const p1; typedef const int* pointer; pointer p1; 相当于是const int* p1;_typedef int* pointer; pointer()

[Vivado 12-1345] Error(s) found during DRC. Bitgen not run.(2018.3版本)_[vivado_tcl 4-78] error(s) found during drc. opt_d-程序员宅基地

文章浏览阅读1.5w次,点赞17次,收藏43次。NSTD #1 Critical Warning 19 out of 19 logical ports use I/O standard (IOSTANDARD) value ‘DEFAULT’, instead of a user assigned specific value. This may cause I/O contention or incompatibility with the ..._[vivado_tcl 4-78] error(s) found during drc. opt_design not run.

【数据结构】:有序树和无序树-程序员宅基地

文章浏览阅读1.3w次,点赞7次,收藏22次。其实有序树和无序树的概念很简单,我们来康康:有序树的定义:若将树中每个结点的各子树看成是从左到右有次序的(即不能互换),则称该树为有序树(Ordered Tree)无序树的定义:若将树中每个结点的各子树从左到右是没有次序的(即可以互换),则称该树为无序树比如我们有这样的例子,在Linux操作系统当中,文件都是以一定的次序排列的,比如在同一个文件夹下的文件,他们一定会以一定的顺序进行排列,比如按照数字的大小顺序或者按照英文字母的顺序进行排列,如下图所示:我们可以..._有序树

软件无线电原理深入解析_gpp sdr技术原理解析-程序员宅基地

文章浏览阅读4k次,点赞9次,收藏41次。SDR技术原理解析 2016/12/03   本博客通俗地介绍SDR的技术原理。先简单介绍SDR的概念,SDR系统的分类,基于GPP的SDR系统。接着详细介绍SDR系统原理,最后再以SDR LTE系统为例进一步解析SDR系统的工作流程。 一 软件无线电(SDR)概念   软件无线电,即Software Defined Radio,SDR。通俗来讲,SDR就是基于通用的硬件平台上用软件来实现各种通信模块。   概念中有两个关键词,“通用硬件平台”和“软件”。..._gpp sdr技术原理解析