`

ThreadLocal类

阅读更多

 ThreadLocal的核心思想很简单:为每个独立的线程提供一个变量的副本。

Java提供的synchronized关键字使用了“同步锁”的机制来阻止线程的竞争访问,即“以时间换空间”。: " 10pt; FONT-SIZE:>    ThreadLocal则使用了“拷贝副本”的方式,人人有份,你用你的,我用我的,大家互不影响,是“以空间换时间”。每个线程修改变量时,实际上修改的是变量的副本,不怕影响到其它线程。

    为了加深对ThreadLocal的理解,下面我使用一个例子来演示ThreadLocal如何隔离线程间的变量访问和修改:

【1】SerialNum类

package example.thread.threadLocal;

public class SerialNum {

    
private static int nextSerialNum = 1;

    @SuppressWarnings(
"unchecked")
    
private static ThreadLocal serialNum = new ThreadLocal() {
        
protected synchronized Object initialValue() {
            return new Integer(nextSerialNum++);      

        }                                                           
    };

    
public static int get() {
        
return ((Integer) (serialNum.get())).intValue();
    }
    
    @SuppressWarnings(
"unchecked")
    
public static void set(Integer newSerial){
        serialNum.set(newSerial);
    }
}


【2】GetSerialNumThread

package example.thread.threadLocal;

public class GetSerialNumThread implements Runnable {

    
public static void main(String args[]) {

        GetSerialNumThread serialNumGetter 
= new GetSerialNumThread();
        Thread t1 
= new Thread(serialNumGetter, "Thread A");
        Thread t2 
= new Thread(serialNumGetter, "Thread B");
        t1.start();
        
try {
            t1.join();
        } 
catch (InterruptedException e) {
            e.printStackTrace();
        }    
        t2.start();            
    }

    
public void run() {
        
int mySerialNum = getSerialNum();
        System.out.println("线程 " + Thread.currentThread().getName()
                
+ " 获取到的序列号是" + mySerialNum);
        System.out.println(
"线程 " + Thread.currentThread().getName()
                
+ " 修改了序列号为" + (mySerialNum * 3));
        setSerialNum(mySerialNum * 3);

        System.out.println(
"线程 " + Thread.currentThread().getName()
                
+ " 再次获得的序列号是" + getSerialNum());
    }

    
private int getSerialNum() {
        
return SerialNum.get();
    }

    
private void setSerialNum(int newSerialNum) {
        SerialNum.set(
new Integer(newSerialNum));
    }
}


运行的结果如下:

线程 Thread A 获取到的序列号是1
线程 Thread A 修改了序列号为3
线程 Thread A 再次获得的序列号是3
线程 Thread B 获取到的序列号是2
线程 Thread B 修改了序列号为6
线程 Thread B 再次获得的序列号是6


    可见第一个线程在调用SerialNum.set(int)方法修改static变量时,其实修改的是它自己的副本,而不是修改本地变量,第二个线程在初始化的时候拿到的序列号是2而不是7。

    为什么会这样呢?明明serialNum是静态变量啊?其实我们只需要看看ThreadLocal的内部构造就知道了:

A. ThreadLocal的get()方法:

 /**
     * Returns the value in the current thread's copy of this thread-local
     * variable.  Creates and initializes the copy if this is the first time
     * the thread has called this method.
     *
     * 
@return the current thread's value of this thread-local
     
*/
    
public T get() {
        Thread t 
= Thread.currentThread();
        ThreadLocalMap map 
= getMap(t);
        
if (map != null)
            
return (T)map.get(this);

        
// Maps are constructed lazily.  if the map for this thread
        
// doesn't exist, create it, with this ThreadLocal and its
        
// initial value as its only entry.
        T value = initialValue();
        createMap(t, value);
        
return value;
    }


B. ThreadLocal的set()方法:

/**
     * Sets the current thread's copy of this thread-local variable
     * to the specified value.  Many applications will have no need for
     * this functionality, relying solely on the {
@link #initialValue}
     * method to set the values of thread-locals.
     *
     * 
@param value the value to be stored in the current threads' copy of
     *        this thread-local.
     
*/
    
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在内部维护了一个Map,将变量的值和线程绑定起来,get/set方法都是对该线程对应的value进行操作,所以不会影响到其它线程。

分享到:
评论

相关推荐

    JAVA ThreadLocal类深入

    【JAVA ThreadLocal类深入】 Java中的ThreadLocal类是一种线程绑定机制,用于在多线程环境中为每个线程提供独立的变量副本,避免了线程间的数据共享带来的并发访问问题。ThreadLocal并不是一个线程对象,而是线程...

    简单分析Java线程编程中ThreadLocal类的使用共

    Java线程编程中的ThreadLocal类是一个非常重要的工具,它在多线程环境下提供了一种线程局部变量的机制。ThreadLocal并非是简单的变量,而是一种能够确保每个线程都拥有独立副本的变量容器。理解ThreadLocal的工作...

    入研究java.lang.ThreadLocal类.docx

    ### 知识点详解:Java.lang.ThreadLocal 类 #### 一、概述 **ThreadLocal** 并非线程的一种特殊实现形式,而是一种为每个线程提供独立副本的机制,通常被称为“线程局部变量”。这种机制使得每个线程都可以独立...

    Java 中ThreadLocal类详解

    Java中的ThreadLocal类是一个强大的工具,它允许程序员创建线程局部变量。这意味着每个线程都有其自己独立的、不可见的变量副本,从而避免了线程间的数据共享问题,简化了多线程环境下的编程复杂性。ThreadLocal并不...

    java中ThreadLocal类的使用

    Java中的`ThreadLocal`类是一个非常实用的工具,它提供了线程局部变量的功能。线程局部变量意味着每个线程都拥有自己独立的变量副本,互不干扰,这在多线程编程中尤其有用,可以避免数据共享带来的同步问题。下面...

    Java ThreadLocal类应用实战案例分析

    Java ThreadLocal类应用实战案例分析 Java ThreadLocal类是Java语言中的一种线程局部变量机制,允许每个线程都拥有自己的变量副本,从而避免了多线程之间的变量冲突。在本文中,我们将通过实战案例分析Java ...

    理解的ThreadLocal类的相关源码(用于博文引用源码下载)

    Java中ThreadLocal工具类(解决多线程程序中并发问题的一种新思路,主要为参数的拷贝问题),感兴趣的话可以查看博文,博文地址:http://blog.csdn.net/otengyue/article/details/38459327

    实例讲解Java并发编程之ThreadLocal类

    Java并发编程中的ThreadLocal类是一个非常重要的工具,它允许我们在多线程环境下为每个线程维护独立的变量副本,从而避免了线程间的数据共享和同步问题。ThreadLocal并不是一个线程,而是一个线程局部变量的容器。...

    快速了解Java中ThreadLocal类

    ThreadLocal类位于`java.lang`包下,其设计目标是为每个线程提供一种便捷的方式,来存储与该线程相关的数据。每个ThreadLocal实例都有一个与之关联的线程局部变量,每个线程都可以通过`get()`方法获取属于自己的变量...

    ThreadLocal的用处

    ThreadLocal类的核心在于`initialValue()`方法和`get()`方法。`initialValue()`是线程首次访问ThreadLocal变量时调用的,返回的是默认值。`get()`方法则会查找当前线程的ThreadLocalMap,如果没有找到对应的值,就...

    Java多线程编程中ThreadLocal类的用法及深入

    在Java多线程编程中,ThreadLocal类是一个非常重要的工具,它主要用于解决线程安全问题。ThreadLocal提供了一个线程局部变量,也就是说,每个线程都有自己的独立副本,互不影响。这种机制使得线程间的数据隔离得以...

    53.线程间的通信-join方法-ThreadLocal类.mp4

    在学习Java过程中,自己收集了很多的Java的学习资料,分享给大家,有需要的欢迎下载,希望对大家有用,一起学习,一起进步。

    理解ThreadLocal

    ThreadLocal 是 Java 中一个非常重要的类,它为解决多线程程序的并发问题提供了一种新的思路。ThreadLocal 并不是一个 Thread,而是 Thread 的局部变量,它为每个使用该变量的线程提供独立的变量副本,所以每一个...

    ThreadLocal源码分析和使用

    ThreadLocal 并不是 Java 的新发明,而是一个变相地通过 ThreadLocal 的类提供支持的解决方案。 ThreadLocal 类中的方法包括: * void set(T value):将此线程局部变量的当前线程副本中的值设置为指定值。 * void ...

    ThreadLocal详解

    在ThreadLocal类的内部,它利用了一个HashMap来存储每个线程的变量副本。在这个Map中,键(Key)是线程对象本身,而值(Value)则是线程的变量副本。由于线程对象作为键是唯一的,因此每个线程都有其对应的独立变量...

    ThreadLocal的简单理解.doc

    ThreadLocal 是 Java 中的一个类,它提供了一个简单的方式来在每个线程中存储变量,并且能够确保这些变量之间不受影响。下面是对 ThreadLocal 的简单理解。 一、背景 最近有人问我 ThreadLocal 是如何做到在每个...

    使用Java ThreadLocal.docx

    创建ThreadLocal变量非常简单,只需实例化ThreadLocal类即可,如下所示: ```java private ThreadLocal<String> myThreadLocal = new ThreadLocal(); ``` 这里的`<String>`是可选的,用于指定ThreadLocal变量所...

    DBUTILs的ThreadLocal

    DbUTils中用ThreadLocal类

    ThreadLocal那点事儿编程开发技术共6页.pdf

    这篇文档深入探讨了Java中的ThreadLocal类,这是一个在多线程编程中非常关键的工具。ThreadLocal提供了线程局部变量,也就是说,每个线程都有自己的独立副本,互不影响。它的工作原理是通过内部的哈希表存储每个线程...

Global site tag (gtag.js) - Google Analytics