本地线程(Threadlocal)
什么是ThreadLocal变量
ThreadLoal 变量,线程局部变量,同一个 ThreadLocal 所包含的对象,在不同的 Thread 中有不同的副本。这里有几点需要注意:
- 因为每个 Thread 内有自己的实例副本,且该副本只能由当前 Thread 使用。这是也是 ThreadLocal 命名的由来。
- 既然每个 Thread 有自己的实例副本,且其它 Thread 不可访问,那就不存在多线程间共享的问题。
- 存储模型是hashMap,key是每个线程的id,值为保存的数据
ThreadLocal 提供了线程本地的实例。它与普通变量的区别在于,每个使用该变量的线程都会初始化一个完全独立的实例副本。ThreadLocal 变量通常被private static修饰。当一个线程结束时,它所使用的所有 ThreadLocal 相对的实例副本都可被回收。
总的来说,ThreadLocal 适用于每个线程需要自己独立的实例且该实例需要在多个方法中被使用,也即变量在线程间隔离而在方法或类间共享的场景。
ThreadLocal实现原理
首先 ThreadLocal 是一个泛型类,保证可以接受任何类型的对象。
因为一个线程内可以存在多个 ThreadLocal 对象,所以其实是 ThreadLocal 内部维护了一个 Map ,这个 Map 不是直接使用的 HashMap ,而是 ThreadLocal 实现的一个叫做 ThreadLocalMap 的静态内部类。而我们使用的 get()、set() 方法其实都是调用了这个ThreadLocalMap类对应的 get()、set() 方法
本地线程设置数据,获取数据
package com.ln.concurrent.chapter10;
/**
* @ProjectName: java-concurrency
* @Package: com.ln.concurrent.chapter10
* @Name:ThreadLocalSimpleTest
* @Author:linianest
* @CreateTime:2020/3/28 21:00
* @version:1.0
* @Description TODO: ThreadLocal线程
*/
public class ThreadLocalSimpleTest {
/**
* threadlocal 通过initialValue设置初始值
*/
private static ThreadLocal<String> threadLocal = new ThreadLocal() {
@Override
protected Object initialValue() {
return "Alex";
}
};
public static void main(String[] args) throws InterruptedException {
// threadLocal.set("Alex");
Thread.sleep(1000);
System.out.println(threadLocal.get());
}
}
多线程访问本地线程
package com.ln.concurrent.chapter10;
import java.util.Arrays;
import java.util.Random;
/**
* @ProjectName: java-concurrency
* @Package: com.ln.concurrent.chapter10
* @Name:ThreadLocalComplexTest
* @Author:linianest
* @CreateTime:2020/3/28 21:40
* @version:1.0
* @Description TODO: 线程ThreadLocal
*/
public class ThreadLocalComplexTest {
private final static ThreadLocal<String> threadlocal = new ThreadLocal<>();
public static final Random random = new