文档章节

ThreadLocal详解

无名猪
 无名猪
发布于 2016/08/08 15:12
字数 824
阅读 48
收藏 2

     ThreadLocal翻译过来是本地线程的意思,但实际意义则不然,更贴切的英文其实应该是ThreadLocalVariable,本地线程变量,就更容易理解了。首先要明确的是ThreadLocal不是线程之间的变量共享,只是线程内部的一个本地变量,就如同源代码中的说法一样:

* This class provides thread-local variables.  These variables differ from
 * their normal counterparts in that each thread that accesses one (via its
 * <tt>get</tt> or <tt>set</tt> method) has its own, independently initialized
 * copy of the variable.  <tt>ThreadLocal</tt> instances are typically private
 * static fields in classes that wish to associate state with a thread (e.g.,
 * a user ID or Transaction ID).

下面就从源码角度详解ThreadLocal。  

     每个Thread都含有一个ThreadLocal.ThreadLocalMap的变量threadLocals,而ThreadLocalMap是一个HashMap的Entry[],而Entry则如下,其中的ThreadLocal是弱引用。

static class Entry extends WeakReference<ThreadLocal> {
            /** The value associated with this ThreadLocal. */
            Object value;

            Entry(ThreadLocal k, Object v) {
                super(k);
                value = v;
            }
        }

ThreadLocal的主要方法主要有get()和set().

 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()方法首先取得当前的线程,然后取得当前线程的ThreadLocalMap,如果map不为null则再取出Entry,为空则初始化ThreadLocalMap并返回ThreadLoca默认值null,这里其实就是延迟初始化的很好的范例,当新建线程时ThreadLocal.ThreadLocalMap threadLocals = null;并没有初始化,而是当真正要使用的时候才进行初始化(如果先调用set()方法也会进行初始化)。

注意取ThreadLocalMap的参数为当前线程t,而取Entry参数则为this。

 public void set(T value) {
        Thread t = Thread.currentThread();
        ThreadLocalMap map = getMap(t);
        if (map != null)
            map.set(this, value);
        else
            createMap(t, value);
    }

set方法也是先取得当前线程的ThreadLocalMap,如果map不为空则将当前threadlocal和value存放到Entry的hashMap中(方法与hashmap的putt方法类似),如果map为空则进程初始化。

从上面的源码和方法可以看出:ThreadLocalMap其实是ThreadLocal类的一个静态内部类,它实现了键值对的设置和获取(键为ThreadLocal,值为value),每个线程中都有一个独立的ThreadLocalMap,它所存储的值,只能被当前线程(currentThread)读取和修改。还有就是,ThreadLocalMap存储的键值对中的键是this对象指向的ThreadLocal对象,而值就是你所设置的对象了,这也就是为什么get和Set方法中为什么既有当前线程,又有this关键字了。跟线程同步,共享变量没有关系,只是线程内部的变量,线程内部使用,不必进行参数传递就可以进行对象访问。

下面就通过一个例子来说明ThredLocal<T>是如何实现线程之间互不干扰的。

package threadLearn;

public class Test {
    ThreadLocal<Long> longLocal = new ThreadLocal<Long>();
      
    public void set() {
        longLocal.set(Thread.currentThread().getId());        
    }
     
    public long getLong() {
        return longLocal.get();
    }
     
   
    public static void main(String[] args) throws InterruptedException {
        final Test test = new Test();

        test.set();
        System.out.println(Thread.currentThread().getName()+":"+test.getLong());    
               
        Thread thread1 = new Thread(){
            public void run() {
               test.set();
               System.out.println(Thread.currentThread().getName()+":"+test.getLong());               
            };
        };
        thread1.start();
        thread1.join();
         
        System.out.println(Thread.currentThread().getName()+":"+test.getLong());           	
    	}
}

执行结果如下:

main:1
Thread-0:9
main:1

前面说过ThreadLocalMap其中的ThreadLocal是弱引用,那会不会引发内存泄露呢?盗用一张图看下就可以一目了然了,在此不详细说明。为了避免发生内存泄露,建议使用remove方法。

 

© 著作权归作者所有

共有 人打赏支持
上一篇: 快速排序
下一篇: HashMap VS Hashtable
无名猪
粉丝 1
博文 17
码字总数 9618
作品 0
南京
程序员
私信 提问

暂无文章

EOS docker开发环境

使用eos docker镜像是部署本地EOS开发环境的最轻松愉快的方法。使用官方提供的eos docker镜像,你可以快速建立一个eos开发环境,可以迅速启动开发节点和钱包服务器、创建账户、编写智能合约....

汇智网教程
今天
8
0
《唐史原来超有趣》的读后感优秀范文3700字

《唐史原来超有趣》的读后感优秀范文3700字: 作者:花若离。我今天分享的内容《唐史原来超有趣》这本书的读后感,我将这本书看了一遍之后就束之高阁了,不过里面的内容一直在在脑海中回放,...

原创小博客
今天
14
0
IC-CAD Methodology知识图谱

CAD (Computer Aided Design),计算机辅助设计,指利用计算机及其图形设备帮助设计人员进行设计工作,这个定义同样可以用来近似描述IC公司CAD工程师这个岗位的工作。 早期IC公司的CAD岗位最初...

李艳青1987
今天
14
0
CompletableFuture get方法一直阻塞或抛出TimeoutException

问题描述 最近刚刚上线的服务突然抛出大量的TimeoutException,查询后发现是使用了CompletableFuture,并且在执行future.get(5, TimeUnit.SECONDS);时抛出了TimeoutException异常,导致接口响...

xiaolyuh
今天
8
0
dubbo 搭建与使用

官网:http://dubbo.apache.org/en-us/ 一,安装监控中心(可以不安装) admin管理控制台,monitor监控中心 下载 bubbo ops 这个是新版的,需要node.js环境,我没有就用老版的了...

小兵胖胖
今天
16
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部