热门标签 | HotTags
当前位置:  开发笔记 > 运维 > 正文

详解SimpleDateFormat的线程安全问题与解决方案

这篇文章主要介绍了SimpleDateFormat的线程安全问题与解决方案,非常不错,具有参考借鉴价值,需要的朋友可以参考下

1. 原因

SimpleDateFormat(下面简称sdf)类内部有一个Calendar对象引用,它用来储存和这个sdf相关的日期信息,例如sdf.parse(dateStr), sdf.format(date) 诸如此类的方法参数传入的日期相关String, Date等等, 都是交友Calendar引用来储存的.这样就会导致一个问题,如果你的sdf是个static的, 那么多个thread 之间就会共享这个sdf, 同时也是共享这个Calendar引用, 并且, 观察 sdf.parse() 方法,你会发现有如下的调用:

Date parse() {
 calendar.clear(); // 清理calendar
 ... // 执行一些操作, 设置 calendar 的日期什么的
 calendar.getTime(); // 获取calendar的时间
}

这里会导致的问题就是, 如果 线程A 调用了 sdf.parse(), 并且进行了 calendar.clear()后还未执行calendar.getTime()的时候,线程B又调用了sdf.parse(), 这时候线程B也执行了sdf.clear()方法, 这样就导致线程A的的calendar数据被清空了(实际上A,B的同时被清空了). 又或者当 A 执行了calendar.clear() 后被挂起, 这时候B 开始调用sdf.parse()并顺利i结束, 这样 A 的 calendar内存储的的date 变成了后来B设置的calendar的date

2. 问题重现

import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;
/**
 * @author zhenwei.liu created on 2013 13-8-29 下午5:35
 * @version $Id$
 */
public class DateFormatTest extends Thread {
 private static SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
 private String name;
 private String dateStr;
 private boolean sleep;
 public DateFormatTest(String name, String dateStr, boolean sleep) {
  this.name = name;
  this.dateStr = dateStr;
  this.sleep = sleep;
 }
 @Override
 public void run() {
  Date date = null;
  if (sleep) {
   try {
    TimeUnit.MILLISECONDS.sleep(2000);
   } catch (InterruptedException e) {
    e.printStackTrace();
   }
  }
  try {
   date = sdf.parse(dateStr);
  } catch (ParseException e) {
   e.printStackTrace();
  }
  System.out.println(name + " : date: " + date);
 }
 public static void main(String[] args) throws InterruptedException {
  ExecutorService executor = Executors.newCachedThreadPool();
  // A 会sleep 2s 后开始执行sdf.parse()
  executor.execute(new DateFormatTest("A", "1991-09-13", true));
  // B 打了断点,会卡在方法中间
  executor.execute(new DateFormatTest("B", "2013-09-13", false));
  executor.shutdown();
 }
}

使用Debug模式执行这段代码,并在sdf.parse()方法里打上断点

parse() {
 calendar.clear()
 // 这里打一个断点
 calendar.getTime()
}

过程:

1) 首先A线程跑起来以后会进入sleep

2) B线程跑起来, 卡在断点处

3) A线程醒过来, 执行 calendar.clear(), 并将设置sdf.calendar的date为1991-09-13, 此时 A B 的 calendar 都为 1991-09-13

4) 让断点继续执行, 输出如下

A : date: Fri Sep 13 00:00:00 CDT 1991
B : date: Fri Sep 13 00:00:00 CDT 1991

 这并不是我们期待的结果

3. 解决方案

最简单的解决方案我们可以把static去掉,这样每个新的线程都会有一个自己的sdf实例,从而避免线程安全的问题

然而,使用这种方法,在高并发的情况下会大量的new sdf以及销毁sdf,这样是非常耗费资源的

在并发情况下,网站的请求任务与线程执行情况大概可以理解为如下

例如Tomcat的线程池的最大Thread数为4, 现在需要执行的任务有1000个(理解为有1000个用户点了你的网站的某个功能),

而这1000个任务都会用到我们写的日期函数处理类

A) 假如说日期函数处理类使用的是new SimpleDateFormat的方法,那么这里就会有1000次sdf的创建和销毁

B) Java中提供了一种ThreadLocal的解决方案,它的工作方式是,每个线程只会有一个实例,也就是说我们执行完这1000个任务,总共只会实例化4个sdf.

而且,它并不会有多线程的并发问题,因为,单个线程执行任务肯定是顺序的,例如Thread #1负责执行Task #1-#250, 那么他是顺序而执行Task #1-#250

而Thread #2拥有自己的sdf实例,他也是顺序执行任务 Task #251-#500, 以此类推

下面是一个使用ThreadLocal解决sdf多线程问题的例子

import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.HashMap;
import java.util.Map;
/**
 * @author zhenwei.liu created on 2013 13-8-29 下午5:35
 * @version $Id$
 */
public class DateUtil {
 /** 锁对象 */
 private static final Object lockObj = new Object();
 /** 存放不同的日期模板格式的sdf的Map */
 private static Map> sdfMap = new HashMap>();
 /**
  * 返回一个ThreadLocal的sdf,每个线程只会new一次sdf
  * 
  * @param pattern
  * @return
  */
 private static SimpleDateFormat getSdf(final String pattern) {
  ThreadLocal tl = sdfMap.get(pattern);
  // 此处的双重判断和同步是为了防止sdfMap这个单例被多次put重复的sdf
  if (tl == null) {
   synchronized (lockObj) {
    tl = sdfMap.get(pattern);
    if (tl == null) {
     // 只有Map中还没有这个pattern的sdf才会生成新的sdf并放入map
     System.out.println("put new sdf of pattern " + pattern + " to map");
     // 这里是关键,使用ThreadLocal替代原来直接new SimpleDateFormat
     tl = new ThreadLocal() {
      @Override
      protected SimpleDateFormat initialValue() {
       System.out.println("thread: " + Thread.currentThread() + " init pattern: " + pattern);
       return new SimpleDateFormat(pattern);
      }
     };
     sdfMap.put(pattern, tl);
    }
   }
  }
  return tl.get();
 }
 /**
  * 是用ThreadLocal来获取SimpleDateFormat,这样每个线程只会有一个SimpleDateFormat
  * 
  * @param date
  * @param pattern
  * @return
  */
 public static String format(Date date, String pattern) {
  return getSdf(pattern).format(date);
 }
 public static Date parse(String dateStr, String pattern) throws ParseException {
  return getSdf(pattern).parse(dateStr);
 }
}

测试类

import java.text.ParseException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;
/**
 * @author zhenyu.nie created on 2013 13-8-26 下午2:13
 * @version 1.0.0
 */
public class Test {
 public static void main(String[] args) {
  final String patten1 = "yyyy-MM-dd";
  final String patten2 = "yyyy-MM";
  Thread t1 = new Thread() {
   @Override
   public void run() {
    try {
     DateUtil.parse("1992-09-13", patten1);
    } catch (ParseException e) {
     e.printStackTrace();
    }
   }
  };
  Thread t2 = new Thread() {
   @Override
   public void run() {
    try {
     DateUtil.parse("2000-09", patten2);
    } catch (ParseException e) {
     e.printStackTrace();
    }
   }
  };
  Thread t3 = new Thread() {
   @Override
   public void run() {
    try {
     DateUtil.parse("1992-09-13", patten1);
    } catch (ParseException e) {
     e.printStackTrace();
    }
   }
  };
  Thread t4 = new Thread() {
   @Override
   public void run() {
    try {
     DateUtil.parse("2000-09", patten2);
    } catch (ParseException e) {
     e.printStackTrace();
    }
   }
  };
  Thread t5 = new Thread() {
   @Override
   public void run() {
    try {
     DateUtil.parse("2000-09-13", patten1);
    } catch (ParseException e) {
     e.printStackTrace();
    }
   }
  };
  Thread t6 = new Thread() {
   @Override
   public void run() {
    try {
     DateUtil.parse("2000-09", patten2);
    } catch (ParseException e) {
     e.printStackTrace();
    }
   }
  };
  System.out.println("单线程执行: ");
  ExecutorService exec = Executors.newFixedThreadPool(1);
  exec.execute(t1);
  exec.execute(t2);
  exec.execute(t3);
  exec.execute(t4);
  exec.execute(t5);
  exec.execute(t6);
  exec.shutdown();
  sleep(1000);
  System.out.println("双线程执行: ");
  ExecutorService exec2 = Executors.newFixedThreadPool(2);
  exec2.execute(t1);
  exec2.execute(t2);
  exec2.execute(t3);
  exec2.execute(t4);
  exec2.execute(t5);
  exec2.execute(t6);
  exec2.shutdown();
 }
 private static void sleep(long millSec) {
  try {
   TimeUnit.MILLISECONDS.sleep(millSec);
  } catch (InterruptedException e) {
   e.printStackTrace();
  }
 }
}

输出

单线程执行:

put new sdf of pattern yyyy-MM-dd to map
thread: Thread[pool-1-thread-1,5,main] init pattern: yyyy-MM-dd
put new sdf of pattern yyyy-MM to map
thread: Thread[pool-1-thread-1,5,main] init pattern: yyyy-MM

双线程执行:

thread: Thread[pool-2-thread-1,5,main] init pattern: yyyy-MM-dd
thread: Thread[pool-2-thread-2,5,main] init pattern: yyyy-MM
thread: Thread[pool-2-thread-1,5,main] init pattern: yyyy-MM
thread: Thread[pool-2-thread-2,5,main] init pattern: yyyy-MM-dd

从输出我们可以看出:

1) 1个线程执行这6个任务的时候,这个线程首次使用过的时候会new一个新的sdf,并且以后都一直用这个sdf,而不是每次处理任务都新建一个新的sdf

2) 2个线程执行6个任务的时候也是同理,但是2个线程的sdf是分开的,每个线程都有自己的"yyyy-MM-dd", "yyyy-MM"的sdf,所以他们不会有线程安全安全问题

试想,如果使用的是new的实现方法,那么不管是用1个线程去执行,还是用2个线程去执行这6个任务,都需要new 6个sdf

以上所述是小编给大家介绍的SimpleDateFormat的线程安全问题与解决方案,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对网站的支持!


推荐阅读
  • 这是原文链接:sendingformdata许多情况下,我们使用表单发送数据到服务器。服务器处理数据并返回响应给用户。这看起来很简单,但是 ... [详细]
  • Java序列化对象传给PHP的方法及原理解析
    本文介绍了Java序列化对象传给PHP的方法及原理,包括Java对象传递的方式、序列化的方式、PHP中的序列化用法介绍、Java是否能反序列化PHP的数据、Java序列化的原理以及解决Java序列化中的问题。同时还解释了序列化的概念和作用,以及代码执行序列化所需要的权限。最后指出,序列化会将对象实例的所有字段都进行序列化,使得数据能够被表示为实例的序列化数据,但只有能够解释该格式的代码才能够确定数据的内容。 ... [详细]
  • 本文介绍了Web学习历程记录中关于Tomcat的基本概念和配置。首先解释了Web静态Web资源和动态Web资源的概念,以及C/S架构和B/S架构的区别。然后介绍了常见的Web服务器,包括Weblogic、WebSphere和Tomcat。接着详细讲解了Tomcat的虚拟主机、web应用和虚拟路径映射的概念和配置过程。最后简要介绍了http协议的作用。本文内容详实,适合初学者了解Tomcat的基础知识。 ... [详细]
  • Tomcat/Jetty为何选择扩展线程池而不是使用JDK原生线程池?
    本文探讨了Tomcat和Jetty选择扩展线程池而不是使用JDK原生线程池的原因。通过比较IO密集型任务和CPU密集型任务的特点,解释了为何Tomcat和Jetty需要扩展线程池来提高并发度和任务处理速度。同时,介绍了JDK原生线程池的工作流程。 ... [详细]
  • 本文讨论了B360主板是否可以安装win7系统的问题。由于B360主板不支持win7系统且缺乏官方驱动的支持,安装win7系统可能存在兼容性和稳定性问题。然而,通过借助USB3.0转接卡,B360主板仍然可以安装win7系统,但USB接口无法使用。相比之下,B365主板可以直接支持win7系统,并提供了相应的驱动,具有更好的稳定性和兼容性。选择合适的主板对于安装win7系统至关重要。 ... [详细]
  • 标题: ... [详细]
  • 本文介绍了如何使用C#制作Java+Mysql+Tomcat环境安装程序,实现一键式安装。通过将JDK、Mysql、Tomcat三者制作成一个安装包,解决了客户在安装软件时的复杂配置和繁琐问题,便于管理软件版本和系统集成。具体步骤包括配置JDK环境变量和安装Mysql服务,其中使用了MySQL Server 5.5社区版和my.ini文件。安装方法为通过命令行将目录转到mysql的bin目录下,执行mysqld --install MySQL5命令。 ... [详细]
  • 本文总结了淘淘商城项目的功能和架构,并介绍了传统架构中遇到的session共享问题及解决方法。淘淘商城是一个综合性的B2C平台,类似京东商城、天猫商城,会员可以在商城浏览商品、下订单,管理员、运营可以在平台后台管理系统中管理商品、订单、会员等。商城的架构包括后台管理系统、前台系统、会员系统、订单系统、搜索系统和单点登录系统。在传统架构中,可以采用tomcat集群解决并发量高的问题,但由于session共享的限制,集群数量有限。本文探讨了如何解决session共享的问题。 ... [详细]
  • Oracle优化新常态的五大禁止及其性能隐患
    本文介绍了Oracle优化新常态中的五大禁止措施,包括禁止外键、禁止视图、禁止触发器、禁止存储过程和禁止JOB,并分析了这些禁止措施可能带来的性能隐患。文章还讨论了这些禁止措施在C/S架构和B/S架构中的不同应用情况,并提出了解决方案。 ... [详细]
  • MySQL语句大全:创建、授权、查询、修改等【MySQL】的使用方法详解
    本文详细介绍了MySQL语句的使用方法,包括创建用户、授权、查询、修改等操作。通过连接MySQL数据库,可以使用命令创建用户,并指定该用户在哪个主机上可以登录。同时,还可以设置用户的登录密码。通过本文,您可以全面了解MySQL语句的使用方法。 ... [详细]
  • 分享css中提升优先级属性!important的用法总结
    web前端|css教程css!importantweb前端-css教程本文分享css中提升优先级属性!important的用法总结微信门店展示源码,vscode如何管理站点,ubu ... [详细]
  • 如何实现JDK版本的切换功能,解决开发环境冲突问题
    本文介绍了在开发过程中遇到JDK版本冲突的情况,以及如何通过修改环境变量实现JDK版本的切换功能,解决开发环境冲突的问题。通过合理的切换环境,可以更好地进行项目开发。同时,提醒读者注意不仅限于1.7和1.8版本的转换,还要适应不同项目和个人开发习惯的需求。 ... [详细]
  • Servlet多用户登录时HttpSession会话信息覆盖问题的解决方案
    本文讨论了在Servlet多用户登录时可能出现的HttpSession会话信息覆盖问题,并提供了解决方案。通过分析JSESSIONID的作用机制和编码方式,我们可以得出每个HttpSession对象都是通过客户端发送的唯一JSESSIONID来识别的,因此无需担心会话信息被覆盖的问题。需要注意的是,本文讨论的是多个客户端级别上的多用户登录,而非同一个浏览器级别上的多用户登录。 ... [详细]
  • 开发笔记:spring boot项目打成war包部署到服务器的步骤与注意事项
    本文介绍了将spring boot项目打成war包并部署到服务器的步骤与注意事项。通过本文的学习,读者可以了解到如何将spring boot项目打包成war包,并成功地部署到服务器上。 ... [详细]
  • 本文介绍了在sqoop1.4.*版本中,如何实现自定义分隔符的方法及步骤。通过修改sqoop生成的java文件,并重新编译,可以满足实际开发中对分隔符的需求。具体步骤包括修改java文件中的一行代码,重新编译所需的hadoop包等。详细步骤和编译方法在本文中都有详细说明。 ... [详细]
author-avatar
手机用户2502873443
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有