SimpleDateFormat,线程不安全,源码分析

1、先看SimpleDateFormat类图

SimpleDateFormat继承了DateFormat
DateFormat中有个成员变量是Calendar,这个先记住,线程不安全主要就是因为这个成员变量

2、看下图SimpleDateFormat的parse方法

这个方法,主要就两步,
第一步是针对入参text做了一系列操作,这里可以省略不看
第二步就是下图代码中的红框部分:calb.establish(calendar),这个入参就是上面说的
SimpleDateFormat的常成员变量

3、看下图,calb.establish(calendar)方法的实现

图中的三步,都不是线程安全的,因为入参cal是SimpleDateFormat的成员变量
当多个线程同时操作一个SimpleDateFormat的时候,SimpleDateFormat中的Calendar属性也就会被多个线程拿到,那么下面这三步就可能因为多个线程同时执行而乱掉

具体测试代码

我们用多个线程,同时操作SimpleDateFormat的parse方法,如果把SimpleDateFormat放在成员变量上,就会大概率报错,因为此时SimpleDateFormat会被多个线程同时使用。
解决方法有很多,最常见的就是在每次用SimpleDateFormat的时候都重新new一个。


import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;public class MainTest {private static ExecutorService executor = Executors.newCachedThreadPool();// 定义SimpleDateFormat为成员变量,不安全private static final SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");// 使用成员变量转换,线程不安全public static String unSafeFormatDate(Date date) throws ParseException {return sdf.format(date);}// 使用成员变量转换,线程不安全public static Date unSafeParse(String strDate) throws ParseException {return sdf.parse(strDate);}// 每次转换都创建一个SimpleDateFormat,线程安全public static String safeFormatDate(Date date) throws ParseException {SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");return sdf.format(date);}// 每次转换都创建一个SimpleDateFormat,线程安全public static Date safeParse(String strDate) throws ParseException {SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");return sdf.parse(strDate);}public static void main(String[] args) {for (int i = 0; i < 5; i++) {executor.execute(new Runnable() {@Overridepublic void run() {try {Date date = new Date();
//                        System.out.println(Thread.currentThread().getName() + ":" + unSafeFormatDate(date));System.out.println(Thread.currentThread().getName() + ":" + unSafeParse("2020-02-02 02:02:02"));} catch (ParseException e) {e.printStackTrace();}}});}}
}

由于线程不安全,常见的错误如下:

Exception in thread "pool-1-thread-3" Exception in thread "pool-1-thread-1" java.lang.NumberFormatException: multiple pointsat sun.misc.FloatingDecimal.readJavaFormatString(FloatingDecimal.java:1890)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:2089)at java.text.SimpleDateFormat.subParse(SimpleDateFormat.java:1869)at java.text.SimpleDateFormat.parse(SimpleDateFormat.java:1514)at java.text.DateFormat.parse(DateFormat.java:364)at com.zs.qunfx.simpleDateFormat.MainTest.unSafeParse(MainTest.java:27)at com.zs.qunfx.simpleDateFormat.MainTest$1.run(MainTest.java:51)at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149)at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624)at java.lang.Thread.run(Thread.java:748)
Exception in thread "pool-1-thread-8" 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:2084)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 com.zs.qunfx.simpleDateFormat.MainTest.unSafeParse(MainTest.java:27)at com.zs.qunfx.simpleDateFormat.MainTest$1.run(MainTest.java:51)at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149)at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624)at java.lang.Thread.run(Thread.java:748)

关于SimpleDateFormat线程不安全的源码分析相关推荐

  1. 【Java 并发编程】线程池机制 ( 线程池执行任务细节分析 | 线程池执行 execute 源码分析 | 先创建核心线程 | 再放入阻塞队列 | 最后创建非核心线程 )

    文章目录 一.线程池执行任务细节分析 二.线程池执行 execute 源码分析 一.线程池执行任务细节分析 线程池执行细节分析 : 核心线程数 101010 , 最大小成熟 202020 , 非核心线 ...

  2. 线程方法notify/notifyAll源码分析

    众所周知,使用notify/notifyAll方法能唤醒wait等待的线程,那么在底层源码究竟做了些什么呢? 本章内容要解决的问题 问题1:notify/nofityAll真的唤醒了线程吗? 问题2: ...

  3. java开启一个线程_【jdk源码分析】java多线程开启的三种方式

    1.继承Thread类,新建一个当前类对象,并且运行其start()方法 1 packagecom.xiaostudy.thread;2 3 /** 4 * @desc 第一种开启线程的方式5 *@a ...

  4. 4hutool源码分析:DateUtil(时间工具类)-格式化时间(万字长文源码分析,学大佬如何写代码)

    技术活,该赏 点赞再看,养成习惯 看本篇文章前,建议先对java源码的日期和时间有一定的了解,如果不了解的话,可以先看这篇文章: 万字博文教你搞懂java源码的日期和时间相关用法 关联文章: huto ...

  5. 源码分析Dubbo系列文章

       本系列文章主要针对Dubbo2.6.2(dubbox2.8.4)版本,从源码的角度分析Dubbo内部的实现细节,加深对Dubbo的各配置参数底层实现原理的理解,更好的指导Dubbo实践,其目录如 ...

  6. 【Netty源码分析摘录】(八)新连接的接入

    文章目录 1.问题 2.检测新连接接入 3.创建客户端 channel 4. 绑定 NioEventLoop 4.1 register0 4.1.1 doRegister() 4.1.2 pipeli ...

  7. Java Review - SimpleDateFormat线程不安全原因的源码分析及解决办法

    文章目录 概述 复现问题 源码分析 How to Fix ? 每次使用时new一个SimpleDateFormat的实例 加锁 使用ThreadLocal 换API - JodaTime or JDK ...

  8. c++ 线程池_JAVA并发编程:线程池ThreadPoolExecutor源码分析

    前面的文章已经详细分析了线程池的工作原理及其基本应用,接下来本文将从底层源码分析一下线程池的执行过程.在看源码的时候,首先带着以下两个问题去仔细阅读.一是线程池如何保证核心线程数不会被销毁,空闲线程数 ...

  9. HDFS源码分析心跳汇报之BPServiceActor工作线程运行流程

    在<HDFS源码分析心跳汇报之数据结构初始化>一文中,我们了解到HDFS心跳相关的BlockPoolManager.BPOfferService.BPServiceActor三者之间的关系 ...

最新文章

  1. Cesium调用天地图的新问题
  2. 我是IT小小鸟 读书笔记
  3. Linux之sed命令
  4. Spark查找某个IP的归属地,二分算法,try{}catch{}的使用,将结果存MySQL数据库
  5. 深入分析Php处理浮点数的问题
  6. ClassPathResource使用简介
  7. java soot_正确执行3个地址代码的SOOT API
  8. 编译安装mysql5.6.36_MySQL5.6.36编译安装
  9. mysql查询特殊符号时_数据库查询中的特殊字符的问题_MySQL
  10. binding.BindingException: Invalid bound statement (not found): xxx → dao接口和mapper.xml映射文件绑定异常
  11. 机器学习中的特征工程总结
  12. Himall商城Api签名帮助类
  13. 网络计算机自动显示,怎么设置电脑断网后自动报警提醒?
  14. 2048小游戏后端的实现
  15. Mac -- 插入移动硬盘后没有显示
  16. 手把手教大家搭建微信公众号查题
  17. 我的三维mandelbulb制作fractal成长之路[续]
  18. Attention中的Q、K、V
  19. 前端踩坑日记 npm install -g ...
  20. matlab离群值算法_什么是离群值如何检测和删除它们对离群值敏感的算法

热门文章

  1. 二叉树先中后序递归遍历与非递归遍历、层次遍历
  2. 在VMware中装Win server 2012配置Hyper-v
  3. 点击按钮,缩放图片(img.width、img.style.width、img.offsetWidth)
  4. 【 Date 对象 参考手册】
  5. mybatis大于小于的转义
  6. Openstack Havana的两个排错过程
  7. 【读书笔记】《框架设计(第2版)CLR Via C#》中两个比较有趣的知识点
  8. 项目:基于以太网通信,单片机作为客户端,接收CAN-Ethernet的十六进制数据
  9. Leetcode--113. 路径总和Ⅱ
  10. Ajax:异步js和xml