近期公司在进行Java开发者的招聘活动,当中有一道面试题是这种:“请简单描写叙述一下ThreadLocal类的作用。” 结果发现有非常多的面试者没有听说过ThreadLocal或者听说过却不知道这个类到底是用来做什么的。

因此这里写一篇博客来介绍一下ThreadLocal这个类。

在我们日常的项目开发中,ThreadLocal并非一个常常使用的类。它很多其它的是被用在诸如Spring,Tomcat或者是Hibernate这些封装了多线程并发的框架或是容器中。

而它的目的也正是为了解决多线程并发訪问共享数据的问题。

 
虽然普通开发者非常少有机会涉及到它。了解ThreadLocal也依旧有助于他们来学习Java并发编程。

通过阅读ThreadLocal的源代码并了解它解决并发问题的思路,开发者能够更好的理解代码中遇到的多线程bug,更不用提那些在项目开发中须要用到多线程编程的开发者了。

因此,不论你是否用到了ThreadLocal类,都非常有必要学习一下它。

 
在我们讨论代码细节之前,先来看看java concurrent in practice中对于多线程并发问题的描写叙述:
     “全部的多线程问题都能够归结为多个线程訪问共享可变状态时的管理问题。

 
这里的状态也就我们说的数据。这句话说明多线程问题必须在下面三个条件都满足的时候才会发生:
1. 拥有多个线程
2. 共享状态
3. 该状态可变
 
假设当中不论什么一个条件没有办法满足。都不会出现多线程问题:
1. 仅仅有单一的线程。 非常显然,这并没有多线程问题。

2. 共享状态不可变。 如果某条数据被多线程共享。然而该数据是不可变数据。那么它便没有多线程问题。

举例来说: Java中的String类型就是不可变的,因此String的共享并不会导致多线程安全问题。

3. 多线程不共享状态。

随意的数据都由某个线程独占,不与其它线程分享,因此也不会出现多线程问题。

 
那么对应的,解决多线程问题的办法有下面几种:
1. 在訪问状态变量时使用同步。

这是最主要的想法,不论什么一本Java多线程编程的书都会具体描写叙述怎样在Java中使用同步,这里不再赘述。

2. 将状态变量改动为不可变的变量。很多新的编程语言,诸如Scala。便是採用这种办法来解决多线程问题的。
3. 避免线程之间共享状态变量。 我们今天讨论的ThreadLocal,便是属于此类解决的方法。
 
刚刚接触ThreadLocal的同学常常会问这样一个问题:“ThreadLocal是线程安全的么?” 这个问题非常难回答,由于当你问这个问题的时候,便默认的觉得ThreadLocal是为了解决多线程之间共享状态的訪问问题的。尽管ThreadLocal的目的正是如此,可是它所採用的办法是“避免多线程之间共享状态”。

既然没有了多线程的共享状态。也就无所谓是否线程安全了。

因此不能简单的说ThreadLocal是否线程安全,这个问题事实上没有意义。

那么ThreadLocal是怎样做到“避免多线程之间的状态共享”的呢?通过在内部维护一个(当前线程 ->对象)的映射表。每一个线程都仅仅能訪问到映射到自己线程的对象。而无法訪问其他线程的对象。通过这样的方法。避免了多线程之间的状态共享,自然也就无所谓线程安全问题了。
 
假设你将某个对象的引用扩散到多个线程中,并将其设置到ThreadLocal里,那么多个线程所指向的便是同一个对象,对它的訪问当然也是有线程安全问题的。从这个角度来讲,ThreadLocal并非线程安全的。

 
换一个角度来描写叙述: TheadLocal并没有真正解决多线程共享状态的安全问题。它仅仅是通过避免状态共享的办法规避了多线程安全问题。

 
我们来看一下ThreadLocal的源代码(JDK1.6):
  1. /**
  2. * Returns the value in the current thread's copy of this
  3. * thread-local variable.  If the variable has no value for the
  4. * current thread, it is first initialized to the value returned
  5. * by an invocation of the {@link #initialValue} method.
  6. *
  7. * @return the current thread's value of this thread-local
  8. */
  9. public T get() {
  10. Thread t = Thread.currentThread();
  11. ThreadLocalMap map = getMap(t);
  12. if (map != null) {
  13. ThreadLocalMap.Entry e = map.getEntry(this);
  14. if (e != null)
  15. return (T)e.value;
  16. }
  17. return setInitialValue();
  18. }
能够看到,当ThreadLocal的get方法被调用时,首先利用当前线程作为key获得了一个map,而这个map便是当前线程专属的,其他线程无法訪问。在从该Map中找到对应的对象并返回。
 
而set方法正好相反:
  1. /**
  2. * Sets the current thread's copy of this thread-local variable
  3. * to the specified value.  Most subclasses will have no need to
  4. * override this method, relying solely on the {@link #initialValue}
  5. * method to set the values of thread-locals.
  6. *
  7. * @param value the value to be stored in the current thread's copy of
  8. *        this thread-local.
  9. */
  10. public void set(T value) {
  11. Thread t = Thread.currentThread();
  12. ThreadLocalMap map = getMap(t);
  13. if (map != null)
  14. map.set(this, value);
  15. else
  16. createMap(t, value);
  17. }

我们来看一个在Hibernate中使用ThreadLocal的样例:

  1. private static ThreadLocal<Connection> connectionHolder
  2. = new ThreadLoca<Connection>() {
  3. public Connection initialValue() {
  4. return DriverManager.getConnection(DB_URL);
  5. }
  6. };
  7. public static Connection getConnection() {
  8. return ConnectionHolder.get();
  9. }

上面的样例是一个最经典的ThreadLocal使用案例: 在单线程中创建一个单例变量。并在程序启动时初始化该单例变量,从而避免在调用每个方法时都要传递该变量。然而该单例可能并非线程安全的,因此,当多线程应用程序在没有互相协作的情况下,能够通过将该单例变量保存到ThreadLocal中。以确保每个线程都拥有属于自己的实例。

在这里,一个更好理解的说法便是:该单例是一个线程内单例,在多线程应用中,每一个线程里都有一个该单例。

通过学习ThreadLocal,我们可以对正确的在项目中使用它。同一时候。也可以帮助我们对多线程编程有一个更深的认识.

(该文同一时候发表在http://mabusyao.iteye.com/blog/2224898)

初识ThreadLocal的更多相关文章

  1. 初识Java ThreadLocal

    转载自:https://www.cnblogs.com/dreamroute/p/5034726.html ThreadLocal翻译成中文比较准确的叫法应该是:线程局部变量. 这个玩意有什么用处,或 ...

  2. ThreadLocal详解,ThreadLocal源码分析,ThreadLocal图解

    本文脉路: 概念阐释 ---->  原理图解  ------> 源码分析 ------>  思路整理  ----> 其他补充. 一.概念阐述. ThreadLocal 是一个为 ...

  3. Java并发编程:ThreadLocal的使用以及实现原理解析

    前言 前面的文章里,我们学习了有关锁的使用,锁的机制是保证同一时刻只能有一个线程访问临界区的资源,也就是通过控制资源的手段来保证线程安全,这固然是一种有效的手段,但程序的运行效率也因此大大降低.那么, ...

  4. ThreadLocal简单理解

    在java开源项目的代码中看到一个类里ThreadLocal的属性: private static ThreadLocal<Boolean> clientMode = new Thread ...

  5. Android线程管理之ThreadLocal理解及应用场景

    前言: 最近在学习总结Android的动画效果,当学到Android属性动画的时候大致看了下源代码,里面的AnimationHandler存取使用了ThreadLocal,激起了我很大的好奇心以及兴趣 ...

  6. Threadlocal使用Case

    Threadlocal能够为每个线程分配一份单独的副本,使的线程与线程之间能够独立的访问各自副本.Threadlocal 内部维护一个Map,key为线程的名字,value为对应操作的副本. /** ...

  7. Android动画效果之初识Property Animation(属性动画)

    前言: 前面两篇介绍了Android的Tween Animation(补间动画) Android动画效果之Tween Animation(补间动画).Frame Animation(逐帧动画)Andr ...

  8. 多线程映射工具——ThreadLocal

    ThreadLocal相当于一个Map<Thread, T>,各线程使用自己的线程对象Thread.currentThread()作为键存取数据,但ThreadLocal实际上是一个包装了 ...

  9. ThreadLocal 工作原理、部分源码分析

    1.大概去哪里看 ThreadLocal 其根本实现方法,是在Thread里面,有一个ThreadLocal.ThreadLocalMap属性 ThreadLocal.ThreadLocalMap t ...

随机推荐

  1. 休假结束,Linus重回内核开发岗位

    在休假反省一个多月之后,Linus Torvalds 又回来了.10 月 22 日爱丁堡举行的欧洲开源峰会上,Linus Torvalds 将与内核维护者们碰头,这是他重新接管Linux内核开发的第一 ...

  2. 紫书 例题 9-4 UVa 116 ( 字典序递推顺序)

    这道题在递推方式和那个数字三角形有一点相像,很容易推出来 但是这道题要求的是字典序,这里就有一个递推顺序的问题 这里用逆推,顺推会很麻烦,为什么呢? 如果顺推的话,最后一行假设有种情况是最小值,那么你 ...

  3. CSUOJ 1651 Weirdo

    1651: Weirdo Time Limit: 5 Sec  Memory Limit: 128 MBSubmit: 40  Solved: 21[Submit][Status][Web Board ...

  4. hdu(2859)——Phalanx(dp)

    题意: 如今有一个n*n的矩阵,然后每一个格子中都有一个字母(大写或小写组成).然后询问你如今最大的对称子矩阵的边长是多少.注意这里的对角线是从左下角到右上角上去的. 思路: 这道题我自己写出了dp的 ...

  5. 转:向IOS设备发送推送通知

    背景 SMS 和 MMS 消息是由无线运营商通过设备的电话号码向特定设备提供的.实现 SMS/MMS 的服务器端应用程序的开发人员必须费大量精力才能与现有的封闭电信基础架构进行交互(其中包括获取电话号 ...

  6. js---12数据类型,数据类型转换,NaN,

    <!DOCTYPE HTML> <html> <head> <meta http-equiv="Content-Type" content ...

  7. javascript进阶教程第一章案例实战

    javascript进阶教程第一章案例实战 一.学习任务 通过几个案例练习回顾学过的知识 通过练习积累JS的使用技巧 二.实例 练习1:删除确认提示框 实例描述: 防止用户小心单击了“删除”按钮,在用 ...

  8. ElasticSearch概述和定义

    福利 => 每天都推送 欢迎大家,关注微信扫码并加入我的4个微信公众号:   大数据躺过的坑      Java从入门到架构师      人工智能躺过的坑         Java全栈大联盟   ...

  9. javaweb一

    JavaWeb就是在服务器端以Java语言为解释运行基础的web程序. php代码要想在服务器端运行,需要在服务器软件(通常是Apache)上加php的解释器,Java也一样,但是这个解释器是Tomc ...

  10. 微信小程序从零开始开发步骤(六)4种页面跳转的方法

    用法:用于页面跳转,相当于html里面的<a></a>标签. API教程:https://mp.weixin.qq.com/debug/wxadoc/dev/component ...