ThreadLocal详解

1.ThreadLocal简介
多线程访问同一个共享变量的时候容易出现并发问题,特别是多个线程对一个变量进行写入的时候,为了保证线程安全,一般使用者在访问共享变量的时候需要进行额外的同步措施才能保证线程安全性。ThreadLocal是除了加锁这种同步方式之外的一种保证一种规避多线程访问出现线程不安全的方法,当我们在创建一个变量后,如果每个线程对其进行访问的时候访问的都是线程自己的变量这样就不会存在线程不安全问题。

ThreadLocal是JDK包提供的,它提供线程本地变量,如果创建一类ThreadLocal变量,那么访问这个变量的每个线程都会有这个变量的一个副本,在实际多线程操作的时候,操作的是自己本地内存中的变量,从而规避了线程安全问题,如下图所示
  ThreadLocal详解
hreadLocal用于保存某个线程共享变量:对于同一个static ThreadLocal,不同线程只能从中get,set,remove自己的变量,而不会影响其他线程的变量。
(1)ThreadLocal.get: 获取ThreadLocal中当前线程共享变量的值。
(2)ThreadLocal.set: 设置ThreadLocal中当前线程共享变量的值。
(3)ThreadLocal.remove: 移除ThreadLocal中当前线程共享变量的值。
(4)ThreadLocal.initialValue: ThreadLocal没有被当前线程赋值时或当前线程刚调用remove方法后调用get方法,返回此方法值。

举例:
ThreadLocal详解
(1)每个人都有一张银行卡
(2)每个人每张卡都有一定的余额。
(3)每个人获取银行卡余额都必须通过该银行的管理系统。
(4)每个人都只能获取自己卡持有的余额信息,他人的不可访问。
ThreadLocal详解
映射到我们要说的ThreadLocal
(1)card类似于Thread
(2)card余额属性,卡号属性等类似于Treadlocal内部属性集合threadLocals
(3)cardManager类似于ThreadLocal管理类
每个线程只能访问自己线程所拥有的属性。每个线程所设置的属性被hreadLocal来管理。

二.ThreadLocal简单使用
下面的例子中,开启两个线程,在每个线程内部设置了本地变量的值,然后调用print方法打印当前本地变量的值。如果在打印之后调用本地变量的remove方法会删除本地内存中的变量,代码如下所示

 1 package test;
 2 
 3 public class ThreadLocalTest {
 4 
 5     static ThreadLocal<String> localVar = new ThreadLocal<>();
 6 
 7     static void print(String str) {
 8         //打印当前线程中本地内存中本地变量的值
 9         System.out.println(str + " :" + localVar.get());
10         //清除本地内存中的本地变量
11         localVar.remove();
12     }
13 
14     public static void main(String[] args) {
15         Thread t1  = new Thread(new Runnable() {
16             @Override
17             public void run() {
18                 //设置线程1中本地变量的值
19                 localVar.set("localVar1");
20                 //调用打印方法
21                 print("thread1");
22                 //打印本地变量
23                 System.out.println("after remove : " + localVar.get());
24             }
25         });
26 
27         Thread t2  = new Thread(new Runnable() {
28             @Override
29             public void run() {
30                 //设置线程1中本地变量的值
31                 localVar.set("localVar2");
32                 //调用打印方法
33                 print("thread2");
34                 //打印本地变量
35                 System.out.println("after remove : " + localVar.get());
36             }
37         });
38 
39         t1.start();
40         t2.start();
41     }
42 }

下面是运行后的结果:
ThreadLocal详解

上一篇:List中remove()方法的陷阱,被坑惨了


下一篇:数据结构与算法分析-表,栈,队列