【Interview】深入理解ThreadLocal源码

2024-05-13 07:58

本文主要是介绍【Interview】深入理解ThreadLocal源码,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

概述

  • ThreadLocal 是一个本地线程副本变量工具类。主要用于将私有线程和该线程存放的副本对象做一个映射,各个线程之间的变量互不干扰。在高并发场景下,可以实现无状态的调用,适用于各个线程不共享变量值的操作。
  • 内部使用静态内部类ThreadLocalMap存储每个线程变量副本的方法,key存储的是当前线程的ThreadLocal对象,value就是当前ThreadLocal对应的线程变量的的副本值。

提供方法

  • T get() 返回此线程局部变量的当前线程副本中的值。
  • protected T initialValue() 返回此线程局部变量的当前线程的“初始值”。线程第一次使用 get() 方法访问变量时将调用此方法,但如果线程之前调用了 set(T) 方法,则不会对该线程再调用 initialValue 方法。通常,此方法对每个线程最多调用一次,但如果在调用 get() 后又调用了 remove(),则可能再次调用此方法。
  • void remove() 移除此线程局部变量当前线程的值。
  • void set(T value) 将此线程局部变量的当前线程副本中的值设置为指定值。

怎嘛使用

public class ThreadLocalTest {ThreadLocal<Integer> threadLocal=new ThreadLocal<Integer>(){//返回此线程局部变量的当前线程的“初始值”。@Overrideprotected Integer initialValue() {return 0;}};//   返回此线程局部变量的当前线程副本中的值。public int get(){//将此线程局部变量的当前线程副本中的值设置为指定值。threadLocal.set(threadLocal.get()+1);return  threadLocal.get();}public static void main(String[] args) {ThreadLocalTest test=new ThreadLocalTest();new Thread(()->{for (int i = 0; i <3 ; i++) {int state=test.get();System.out.println(Thread.currentThread().getName()+"获取值:"+state);}}).start();new Thread(()->{for (int i = 0; i <3 ; i++) {int state=test.get();System.out.println(Thread.currentThread().getName()+"获取值:"+state);}}).start();}
}//输出Thread-0获取值:1
Thread-0获取值:2
Thread-0获取值:3
Thread-1获取值:1
Thread-1获取值:2
Thread-1获取值:3

源码分析

set()方法

    public void set(T value) {//记录当前线程Thread t = Thread.currentThread();//获取当前线》的ThreadLocalMap ThreadLocalMap map = getMap(t);if (map != null)//ThreadLocalMap  不为空则直接设置当前变成的副本值,map.set(this, value);else//创建ThreadLocalMap  key当前线程对象,value:副本值createMap(t, value);}

ThreadLocalMap 内部类

    static class ThreadLocalMap {static class Entry extends WeakReference<ThreadLocal<?>> {/** 与此ThreadLocal关联的值.  */Object value;Entry(ThreadLocal<?> k, Object v) {super(k);value = v;}}
  • 源码中可以看出 ThreadLocalMap 依靠Entry 来存储ThreadLocal和副本值,key就是ThreadLocalvalue就是ThreadLocal的变量副本值。Entry 集成WeakReference,说明是一个弱引用关系。当一个对象仅仅被弱引用指向, 而没有任何其他强引用指向的时候, 如果这时GC运行, 那么这个对象就会被回收,不论当前的内存空间是否足够,这个对象都会被回收。

getMap方法

	//获取与ThreadLocal关联的Thread中的ThreadLocal。ThreadLocalMap getMap(Thread t) {return t.threadLocals;}
  • ThreadLocal是包含在Thread类中的

ThreadLocalMapset方法

        private void set(ThreadLocal<?> key, Object value) {Entry[] tab = table;int len = tab.length;//计算ThreadLocal 散列值 找到存储位置int i = key.threadLocalHashCode & (len-1);//利用线性探测法找到合适的存储位置for (Entry e = tab[i];e != null;e = tab[i = nextIndex(i, len)]) {ThreadLocal<?> k = e.get();//如果找到的k和传入的key相等,说明存在,覆盖更新即可if (k == key) {e.value = value;return;}// key == null,但是存在值(因为此处的e != null),说明之前的ThreadLocal对象已经被回收了if (k == null) {// 替换之前的的元素replaceStaleEntry(key, value, i);return;}}//不存在对应key的实例,则创建一个新的tab[i] = new Entry(key, value);//增加容量大小int sz = ++size;//        // 如果没有清理陈旧的 Entry 并且数组中的元素大于了阈值,则进行 rehashif (!cleanSomeSlots(i, sz) && sz >= threshold)//整表格的大小。 首先扫描整个表,删除过时的条目。 如果这不足以缩小表的大小,则将表大小加倍。rehash();}

get()操作

    public T get() {// 记录当前访问线程Thread t = Thread.currentThread();//获取当前线程的ThreadLocalMap对象//Thread的    ThreadLocal.ThreadLocalMap threadLocals参数ThreadLocalMap map = getMap(t);if (map != null) {//存在ThreadLocalMap 则获取相对应的EntryThreadLocalMap.Entry e = map.getEntry(this);if (e != null) {@SuppressWarnings("unchecked")//ThreadLocalMap内部类中有一个Entry内部类//依靠Entry`来存储`ThreadLocal`和副本值。直接以ThreadLocal为key获取副本值T result = (T)e.value;return result;}}return setInitialValue();}//getEntry方法private Entry getEntry(ThreadLocal<?> key) {//计算ThreadLocal的在数组中的位置,采用了开放定址法int i = key.threadLocalHashCode & (table.length - 1);Entry e = table[i];//存在则返回if (e != null && e.get() == key)      return e;else//不在在的操作return getEntryAfterMiss(key, i, e);}/**key:线程本地对象i:哈希表的索引e: 对应的Entry*/private Entry getEntryAfterMiss(ThreadLocal<?> key, int i, Entry e) {Entry[] tab = table;int len = tab.length;while (e != null) {ThreadLocal<?> k = e.get();if (k == key)return e;if (k == null)//key == null,有利于GC回收,能够有效地避免内存泄漏。expungeStaleEntry(i);elsei = nextIndex(i, len);e = tab[i];}return null;}

参考:http://cmsblogs.com/?p=2442

这篇关于【Interview】深入理解ThreadLocal源码的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/985123

相关文章

深入理解Mysql OnlineDDL的算法

《深入理解MysqlOnlineDDL的算法》本文主要介绍了讲解MysqlOnlineDDL的算法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小... 目录一、Online DDL 是什么?二、Online DDL 的三种主要算法2.1COPY(复制法)

java 恺撒加密/解密实现原理(附带源码)

《java恺撒加密/解密实现原理(附带源码)》本文介绍Java实现恺撒加密与解密,通过固定位移量对字母进行循环替换,保留大小写及非字母字符,由于其实现简单、易于理解,恺撒加密常被用作学习加密算法的入... 目录Java 恺撒加密/解密实现1. 项目背景与介绍2. 相关知识2.1 恺撒加密算法原理2.2 Ja

Nginx屏蔽服务器名称与版本信息方式(源码级修改)

《Nginx屏蔽服务器名称与版本信息方式(源码级修改)》本文详解如何通过源码修改Nginx1.25.4,移除Server响应头中的服务类型和版本信息,以增强安全性,需重新配置、编译、安装,升级时需重复... 目录一、背景与目的二、适用版本三、操作步骤修改源码文件四、后续操作提示五、注意事项六、总结一、背景与

Android实现图片浏览功能的示例详解(附带源码)

《Android实现图片浏览功能的示例详解(附带源码)》在许多应用中,都需要展示图片并支持用户进行浏览,本文主要为大家介绍了如何通过Android实现图片浏览功能,感兴趣的小伙伴可以跟随小编一起学习一... 目录一、项目背景详细介绍二、项目需求详细介绍三、相关技术详细介绍四、实现思路详细介绍五、完整实现代码

深入解析C++ 中std::map内存管理

《深入解析C++中std::map内存管理》文章详解C++std::map内存管理,指出clear()仅删除元素可能不释放底层内存,建议用swap()与空map交换以彻底释放,针对指针类型需手动de... 目录1️、基本清空std::map2️、使用 swap 彻底释放内存3️、map 中存储指针类型的对象

深入理解go中interface机制

《深入理解go中interface机制》本文主要介绍了深入理解go中interface机制,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学... 目录前言interface使用类型判断总结前言go的interface是一组method的集合,不

深入解析Java NIO在高并发场景下的性能优化实践指南

《深入解析JavaNIO在高并发场景下的性能优化实践指南》随着互联网业务不断演进,对高并发、低延时网络服务的需求日益增长,本文将深入解析JavaNIO在高并发场景下的性能优化方法,希望对大家有所帮助... 目录简介一、技术背景与应用场景二、核心原理深入分析2.1 Selector多路复用2.2 Buffer

Java Spring的依赖注入理解及@Autowired用法示例详解

《JavaSpring的依赖注入理解及@Autowired用法示例详解》文章介绍了Spring依赖注入(DI)的概念、三种实现方式(构造器、Setter、字段注入),区分了@Autowired(注入... 目录一、什么是依赖注入(DI)?1. 定义2. 举个例子二、依赖注入的几种方式1. 构造器注入(Con

深入理解Go语言中二维切片的使用

《深入理解Go语言中二维切片的使用》本文深入讲解了Go语言中二维切片的概念与应用,用于表示矩阵、表格等二维数据结构,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起学习学习吧... 目录引言二维切片的基本概念定义创建二维切片二维切片的操作访问元素修改元素遍历二维切片二维切片的动态调整追加行动态

从原理到实战深入理解Java 断言assert

《从原理到实战深入理解Java断言assert》本文深入解析Java断言机制,涵盖语法、工作原理、启用方式及与异常的区别,推荐用于开发阶段的条件检查与状态验证,并强调生产环境应使用参数验证工具类替代... 目录深入理解 Java 断言(assert):从原理到实战引言:为什么需要断言?一、断言基础1.1 语