- 浏览: 288656 次
- 性别:
- 来自: 北京
文章分类
最新评论
-
jj19900703:
多个pattern代码如下:public class Date ...
ThreadLocal示例 -
jj19900703:
上面这些评论真是搞笑,根本没弄懂瞎评论误导人。第一:Simpl ...
ThreadLocal示例 -
myumen:
helong0904 写道在世界的中心呼喚愛 写道helong ...
ThreadLocal示例 -
helong0904:
在世界的中心呼喚愛 写道helong0904 写道zhang3 ...
ThreadLocal示例 -
myumen:
在世界的中心呼喚愛 写 ...
ThreadLocal示例
一、ThreadLocal概述
学习JDK中的类,首先看下JDK API对此类的描述,描述如下:
API表达了下面几种观点:
1、ThreadLocal不是线程,是线程的一个变量,你可以先简单理解为线程类的属性变量。
2、ThreadLocal 在类中通常定义为静态类变量。
3、每个线程有自己的一个ThreadLocal,它是变量的一个‘拷贝’,修改它不影响其他线程。
既然定义为类变量,为何为每个线程维护一个副本(姑且成为‘拷贝’容易理解),让每个线程独立访问?多线程编程的经验告诉我们,对于线程共享资源(你可以理解为属性),资源是否被所有线程共享,也就是说这个资源被一个线程修改是否影响另一个线程的运行,如果影响我们需要使用synchronized同步,让线程顺序访问。
ThreadLocal适用于资源共享但不需要维护状态的情况,也就是一个线程对资源的修改,不影响另一个线程的运行;这种设计是‘空间换时间’,synchronized顺序执行是‘时间换取空间’。
二、ThreadLocal方法介绍
T |
get() 返回此线程局部变量的当前线程副本中的值。 |
protected T |
initialValue() 返回此线程局部变量的当前线程的“初始值”。 |
void |
remove() 移除此线程局部变量当前线程的值。 |
void |
set(T value) 将此线程局部变量的当前线程副本中的值设置为指定值。 |
三、深入源码
ThreadLocal有一个ThreadLocalMap静态内部类,你可以简单理解为一个MAP,这个‘Map’为每个线程复制一个变量的‘拷贝’存储其中。
当线程调用ThreadLocal.get()方法获取变量时,首先获取当前线程引用,以此为key去获取响应的ThreadLocalMap,如果此‘Map’不存在则初始化一个,否则返回其中的变量,代码如下:
public T get() { Thread t = Thread.currentThread(); ThreadLocalMap map = getMap(t); if (map != null) { ThreadLocalMap.Entry e = map.getEntry(this); if (e != null) return (T)e.value; } return setInitialValue(); }
调用get方法如果此Map不存在首先初始化,创建此map,将线程为key,初始化的vlaue存入其中,注意此处的initialValue,我们可以覆盖此方法,在首次调用时初始化一个适当的值。setInitialValue代码如下:
private T setInitialValue() { T value = initialValue(); Thread t = Thread.currentThread(); ThreadLocalMap map = getMap(t); if (map != null) map.set(this, value); else createMap(t, value); return value; }
set方法相对比较简单如果理解以上俩个方法,获取当前线程的引用,从map中获取该线程对应的map,如果map存在更新缓存值,否则创建并存储,代码如下:
public void set(T value) { Thread t = Thread.currentThread(); ThreadLocalMap map = getMap(t); if (map != null) map.set(this, value); else createMap(t, value); }
对于ThreadLocal在何处存储变量副本,我们看getMap方法:获取的是当前线程的ThreadLocal类型的threadLocals属性。显然变量副本存储在每一个线程中。
/** * 获取线程的ThreadLocalMap 属性实例 */ ThreadLocalMap getMap(Thread t) { return t.threadLocals; }
上面我们知道变量副本存放于何处,这里我们简单说下如何被java的垃圾收集机制收集,当我们不在使用是调用set(null),此时不在将引用指向该‘map’,而线程退出时会执行资源回收操作,将申请的资源进行回收,其实就是将属性的引用设置为null。这时已经不在有任何引用指向该map,故而会被垃圾收集。
四、ThreadLocal应用示例
在我的另一篇文章,对ThreadLocal的使用做了一个实例,此示例也可以用作生产环境,请参见:http://ari.iteye.com/blog/757641
如有问题请留言讨论,谢谢
评论
ThreadLocal最好不要做成static的
ThreadLocal最好不要做成static的 : 你。。。说反了吧
ThreadLocal最好不要做成static的
<div class="quote_div">
<div class="quote_title">snake1987 写道</div>
<div class="quote_div">其实哪有这么复杂,就一个2层的map<br>第一层是Thread.currentThreand()为key,存的map<br>第二层是ThreadLocal.this为key,存的值,不就取出来了</div>
<p> 这样说也不对,你所说的第一层map是不存在的,你所说的第二层Map实际上是Thread的一个属性,仔细读读<span>niumd</span>的这个这个帖子你就明白了。看看Thread中的一小段代码:</p>
<p> </p>
<pre name="code" class="java"> /* ThreadLocal values pertaining to this thread. This map is maintained
* by the ThreadLocal class. */
ThreadLocal.ThreadLocalMap threadLocals = null;</pre>
<p> 上面的代码来自JDK源码,给Thread定义了一个<span style="font-size: 14.4px; white-space: pre;">ThreadLocal.ThreadLocalMap类型的属性,用于存线程变量。你再看看ThreadLocal中的get和set方法你就明白了!</span></p>
<p> </p>
<p> </p>
</div>
<p><br>这个两层的Map<br>ThreadLocal对象本身存放在线程栈上,他用来存储线程的变量, 可自身也是线程的一个变量? </p>
<p>上一层的大MAP存在堆上?</p>
ThreadLocal不是用来解决对象共享访问问题的,而主要是提供了保持对象的方法和避免参数传递的方便的对象访问方式。
http://www.iteye.com/topic/617368
robbin曾说过
robbin是谁??
robbin 你都不知道,你就不算来过javaeye
static class ThreadLocalMap { /** * The entries in this hash map extend WeakReference, using * its main ref field as the key (which is always a * ThreadLocal object). Note that null keys (i.e. entry.get() * == null) mean that the key is no longer referenced, so the * entry can be expunged from table. Such entries are referred to * as "stale entries" in the code that follows. */ static class Entry extends WeakReference<ThreadLocal> { /** The value associated with this ThreadLocal. */ Object value; Entry(ThreadLocal k, Object v) { super(k); value = v; } }
请看以上的源代码,在ThreadLocalMap 内部的Entry 是WeakReference。 请问这个entry定义为弱类型,不是会很快随机的被GC回收吗? 不过通常一个 Thread的生存时间可能更小,所以应该没有问题,是吧?请GS解答一下
Thread t = Thread.currentThread();
ThreadLocalMap map = t.threadLocals;
这个map的Key为ThreadLocal对象,Value为当前设置的值
设置值:
map.set(ThreadLocal对象, Value);
取值:
map.get(ThreadLocal对象)
<div class="quote_div">其实哪有这么复杂,就一个2层的map<br>第一层是Thread.currentThreand()为key,存的map<br>第二层是ThreadLocal.this为key,存的值,不就取出来了</div>
<p> 这样说也不对,你所说的第一层map是不存在的,你所说的第二层Map实际上是Thread的一个属性,仔细读读<span style="">niumd</span>的这个这个帖子你就明白了。看看Thread中的一小段代码:</p>
<p>
</p>
<pre name="code" class="java"> /* ThreadLocal values pertaining to this thread. This map is maintained
* by the ThreadLocal class. */
ThreadLocal.ThreadLocalMap threadLocals = null;</pre>
<p> 上面的代码来自JDK源码,给Thread定义了一个<span style="font-size: 14.4px; white-space: pre;">ThreadLocal.ThreadLocalMap类型的属性,用于存线程变量。你再看看ThreadLocal中的get和set方法你就明白了!</span></p>
<p> </p>
<p> </p>
自己看看API反而是最好的捷径
robbin曾说过
robbin是谁??
robbin曾说过
在信息化时代 对于网页开发 我们平常所说的性能 不就是指的时间吗?
不禁让我想到索引 索引就是用空间换时间的典型应用 如果大家熟悉lucence的话 大家
应该都深有同感
主要讨论ThreadLocal内部是实现细节,并非简单的JDK翻译;
第一层是Thread.currentThreand()为key,存的map
第二层是ThreadLocal.this为key,存的值,不就取出来了
如果在set一次,再get的时候,拿出来的是第二次的吧,第一个对象自动销毁?
ThreadLocal利用系统的WeakReferences进行对象的维护,
理论上说不需要显式的remove,
但要注意WeakReferences有时也会带起Memory Leaks。
比如inner class的情况下采用ThreadLocal
http://crazybob.org/2006/02/threadlocal-memory-leak.html
等到ThreadLocal set对象到当前的Thread的这个Thread结束了,对象就自动销毁,是吧?
可以set(null);
等到ThreadLocal set对象到当前的Thread的这个Thread结束了,对象就自动销毁,是吧?
如果在set一次,再get的时候,拿出来的是第二次的吧,第一个对象自动销毁?
发表评论
-
Java8学习笔记:Lambda表达式定义
2016-12-03 22:17 21Lambda定义 Lambda是可传递的匿名函数,它没有名 ... -
浅析BlockingQueue实现
2013-05-21 22:16 1640BlockingQueue BlockingQueue ... -
Jboss Netty源码分析一
2011-05-14 10:29 8502本文采用版 ... -
浅谈Selector创建机制
2011-05-10 14:33 1404前段时间阅读mina源码时,理解Selector实例化机制细 ... -
探究Struts2运行机制:StrutsPrepareAndExecuteFilter 源码剖析
2010-11-30 23:29 6343作者:niumd blog:http://ari.i ... -
【翻译】Spring Integration参考手册中文版
2010-11-26 14:36 6443工作主要基于商业中间件TIBCO做EA ... -
ThreadLocal示例
2010-09-07 16:27 26089本文借花献佛,引用Tim Cull的博文“Simple ... -
八个改善Java遗留系统的技巧
2010-09-06 09:32 562你没看错,就是这个题目:即使是Java系统也会变成“ ... -
高并发下NIO socket消息超时机制的探讨
2010-09-02 12:31 2737去年参与项目的 ... -
项目重构之命令模式
2010-06-13 10:23 1035项目中有个业务处理类大小117K,代码2700行,看此 ... -
Java TCP/IP Socket编程
2010-06-11 13:00 592最想想写一篇关于java socket编程的文章,特此占位,提 ... -
Spring JDBC Framework详解——批量JDBC操作、ORM映射
2010-03-17 21:30 3631作者:niumd,转载请注 ... -
翻译:Apache MINA User Guide】Chapter 2 - Basics 之客户端应用程序
2009-12-28 08:40 1066作者:niumd blog:http://ari.iteye ... -
【翻译:Apache MINA User Guide】Chapter 2 - Basics
2009-12-26 15:08 1210本章节我们将详细介绍基于Apache MINA的C/S ... -
【翻译:Apache MINA User Guide 】 Chapter 1
2009-12-26 14:41 1045作者:niumd blog:http://ari.iteye ... -
MyEclipse8 GA 下载地址 注册码 优化指南
2009-12-26 10:51 3676官方网站 ... -
基本类型,字符串转字节数组
2009-09-22 22:39 778接触SMPP协议,SMPP协议中的消息分为消息头、消 ... -
byte与16进制互转
2009-09-22 22:27 1090private static final char[] kD ... -
jxl导出excel:java.lang.ArrayIndexOutOfBoundsExceptio
2009-05-14 18:54 4101先大致说下项目:公司的一个项目小功能需要上传16 ... -
设计模式精解【中文版书签版】
2009-04-15 12:43 1308工作之余研究java设计模式、个人感觉设计模式精解这 ...
相关推荐
ThreadLocal入门教程。 讲解了线程安全和ThreadLocal的使用的基本知识。
ThreadLocal
理解ThreadLocal 理解ThreadLocal 理解ThreadLocal 理解ThreadLocal
ThreadLocal应用示例及理解,这个写了相关的示例,可以参考一下。
本文会基于实际场景介绍ThreadLocal如何使用以及内部实现机制。具有很好的参考价值,下面跟着小编一起来看下吧
深入研究java.lang.ThreadLocal类。ThreadLocal是什么呢?其实ThreadLocal并非是一个线程的本地实现版本,它并不是一个Thread,而是 threadlocalvariable(线程局部变量)。也许把它命名为ThreadLocalVar更加合适。
学习ThreadLocal,了解其中的原理,以及学习其中的优点!避免坑点!!
正确理解ThreadLocal.pdf
DbUTils中用ThreadLocal类
主要介绍ThreadLocal的原理,实例分析以及注意事项
java 简单的ThreadLocal示例
ThreadLocal的几种误区ThreadLocal的几种误区ThreadLocal的几种误区
Synchronized与ThreadLocal
本例以序列号生成的程序为例,展示ThreadLocal的使用
设计模式及ThreadLocal详细讲解资料,想要学习java或者提升自己技术的同学可以下载观看
ThreadLocal保证一个类的实例变量在各个线程中都有一份单独的拷贝, 从而不会影响其他线程中的实例变量
详解java底层实现原理,ThreadLocal底层实现的数据结构,为什么不会导致内存泄露
JDBC事务的封装和Threadlocal实例,参考博客:http://blog.csdn.net/daijin888888/article/details/50988053
ThreadLocal源码分析,主要有ThreadLocal源码以及ThreadLocal的内部结构在jdk8前后的变化