`
niumd
  • 浏览: 288900 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

深入浅出ThreadLocal

    博客分类:
  • java
阅读更多

一、ThreadLocal概述

       学习JDK中的类,首先看下JDK API对此类的描述,描述如下:

JDK API 写道
该类提供了线程局部 (thread-local) 变量。这些变量不同于它们的普通对应物,因为访问某个变量(通过其 get 或 set 方法)的每个线程都有自己的局部变量,它独立于变量的初始化副本。ThreadLocal 实例通常是类中的 private static 字段,它们希望将状态与某一个线程(例如,用户 ID 或事务 ID)相关联。

    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

 

 

如有问题请留言讨论,谢谢

分享到:
评论
2 楼 hoorace 2010-09-08  
ThreadLocal 系统是不会GC的,你到remove方法在神马地方?
1 楼 xiaoxiaoniao 2010-09-08  
非常好的总结。
其核心思想就是用Map来管理局部变量的副本,是这样?

相关推荐

    深入浅出的学习Java ThreadLocal

    Java ThreadLocal 是一个非常重要的工具类,它提供了一种在多线程环境下为每个线程维护独立变量副本的机制。这种机制使得各个线程能够拥有自己的变量实例,而不会互相干扰,降低了数据共享的复杂性。 ### 应用场景 ...

    深入浅出ReentrantReadWriteLock源码解析.docx

    【深入浅出ReentrantReadWriteLock源码解析】 ReentrantReadWriteLock是Java并发包中的一个核心类,它提供了读写锁的实现,使得多个线程可以并发读取共享资源,但写操作是互斥的,即同一时间只能有一个线程进行写...

    浅谈Java引用和Threadlocal的那些事

    浅谈Java引用和Threadlocal的那些事 本文主要介绍了Java引用和Threadlocal的知识点,包括Java中的引用类型、Threadlocal的使用等。 Java中的引用类型: Java中有四种引用类型:强引用(Strong Reference)、软...

    java并发编程实践笔记资料.pdf

    16. 使用ThreadLocal变量可以确保线程封闭性,例如hibernate openSessionInView机制和jdbc的connection机制。 17. 单一不可变对象往往是线程安全的,但是复杂不可变对象需要保证其内部成员变量也是不可变的。 18. ...

    【Java入门知识图谱】帮助Java初学者成长

    【对线面试官】深入浅出Java内存模型 Java虚拟机 【对线面试官】Java从编译到执行,发生了什么? 【对线面试官】双亲委派机制 【对线面试官】JVM内存结构 【对线面试官】垃圾回收机制 【对线面试官】CMS垃圾回收器 ...

    java_concurrency_in_practice_source源代码

    《Java并发编程实践》这本书深入浅出地探讨了Java平台上的并发问题,提供了许多实用的解决方案。这里的"java_concurrency_in_practice_source"源代码正是书中实例的实现,它涵盖了Java多线程编程中的关键概念和技术...

    张孝祥java就业面试宝典

    首先,书中深入浅出地讲解了Java基础,包括语法特性、数据类型、控制结构、类与对象、封装、继承、多态等面向对象编程的基础知识。对于初学者来说,这是理解Java编程思维的重要部分,同时也是面试官考察候选人基础...

    3_Java_Concurrency_in_Practice_proglib_java_practice_

    这本书深入浅出地探讨了在Java环境中如何有效地编写并发程序,涵盖了从基础概念到高级策略的广泛主题,旨在帮助开发者避免并发编程中的陷阱并充分利用多核处理器的优势。 并发编程是现代软件开发中的关键技能,尤其...

    Java开发实战经典(李兴华老师)

    这本书以其深入浅出的讲解方式和丰富的实战案例,为学习者提供了全面而细致的Java编程指导。在这里,我们主要探讨书中的核心知识点以及李老师的笔记所涵盖的详细内容。 首先,Java开发的基础部分是不可或缺的,包括...

    C# 关于多线程开发的经典书籍 并发编程经典实例

    《C#并发编程经典实例》是一本专注于C#多线程开发的重要著作,它深入浅出地探讨了在C#环境中进行并发编程的各种技术和实践。并发编程是现代软件开发中的核心概念,尤其在多核处理器普及后,充分利用系统资源、提高...

    良葛格java jdk 5.0学习笔记

    《良葛格Java JDK 5.0学习笔记》是一份专为Java初学者及爱好者精心编写的教程,它深入浅出地介绍了Java编程语言的核心概念和技术。这份教材以JDK 5.0版本为基础,该版本是Java发展史上的一个重要里程碑,引入了许多...

    Java并发编程实践

    总之,《Java并发编程实践》是一本深入浅出的教程,涵盖了Java并发编程的各个方面,对于任何希望提升并发编程能力的Java开发者来说,都是一本不可或缺的参考书。通过学习书中的知识点,读者能够更好地理解和解决实际...

    Thinking in Java

    这本书深入浅出地讲解了Java语言的核心概念、语法和编程思想,旨在帮助读者建立起扎实的Java基础,并理解面向对象编程的核心理念。 在《Thinking in Java》的第三版中,作者详细阐述了以下知识点: 1. **Java基础*...

    thinking in java 4RD 课后习题答案以及PDF

    这本书深入浅出地讲解了Java语言的核心概念和机制,涵盖了从基础语法到高级特性的全面内容。课后习题是学习过程中的重要部分,它们能够帮助读者巩固所学知识,提升编程能力。这个压缩包提供的就是《Thinking in Java...

    Thinking in Java 3

    这本书深入浅出地讲解了Java编程语言的核心概念和技术,帮助读者建立起坚实的基础,并引导他们以面向对象的思维方式来思考问题。第三版是在前两版的基础上进行了更新和完善,适应了Java语言的发展变化。 在这一版中...

    Core Java Vol 1 & Vol 2

    它深入浅出地讲解了Java语言的核心概念和技术,覆盖了从基础语法到高级特性的全面内容。这本书分为两卷,分别对应Java的基础部分和高级部分。 第一卷主要涵盖了以下知识点: 1. **Java入门**:介绍Java开发环境的...

    Thinking.In.Java.Third.Edition.CHS.zip

    这本书是许多程序员入门Java语言的重要参考书,它深入浅出地讲解了Java编程的核心概念和技术。 在Java编程中,首要的知识点是基础语法,包括变量、数据类型、运算符、流程控制(如if语句、for循环、while循环)以及...

    Thinking in Java, 2nd edition

    这本书深入浅出地介绍了Java编程语言的核心概念和技术,帮助读者建立起扎实的面向对象编程思想。 1. 面向对象编程基础: - 类与对象:Java中的所有事物都是对象,类是创建对象的蓝图。书中详细讲解了如何定义类,...

    疯狂JAVA讲义(第3版)随书光盘

    这本书深入浅出地介绍了Java编程语言的核心概念和技术,为读者提供了丰富的学习资源,包括随书附带的光盘。光盘中包含了全书的教学案例、练习题以及可能的解决方案,旨在帮助读者通过实践加深对Java的理解。 本书的...

    Java Threads

    这本书深入浅出地讲解了Java平台上的多线程编程技术,是Java开发者深入理解并发编程的必备参考资料。在Java编程中,线程是执行流程的基本单元,它允许程序同时执行多个任务,极大地提高了应用程序的效率和响应性。 ...

Global site tag (gtag.js) - Google Analytics