热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

JavaThreadLocal使用和源码分析

文章目录ThreadLocal使用和源码分析概述基本使用源码分析ThreadThreadLocalThreadLocalMap内存泄漏问题ThreadLocal使用和源码分析概述


文章目录

  • ThreadLocal使用和源码分析
    • 概述
    • 基本使用
    • 源码分析
      • Thread
      • ThreadLocal
      • ThreadLocalMap
    • 内存泄漏问题


ThreadLocal使用和源码分析


概述


  • ThreadLocal即线程局部变量。
  • ThreadLocal提供线程的局部变量,每个线程可以通过了get/set方法对局部变量进行操作。
  • 局部变量不会和其他线程的局部变量有冲突,实现了线程的数据隔离。
  • Thread和ThreadLocal关系:
    • Thread内部持有ThreadLocalMap类型的成员变量threadLocals。
    • ThreadLocal是负责操作线程局部变量的管理类,有get()、set()、remove()方法负责获取、设置、清除数据。
    • ThreadLocalMap是ThreadLocal的内部类,用于存储线程的局部变量,内部维护一个Entry数组,键值对结构,键为ThreadLocal类型,值为Object类型。

在这里插入图片描述

在这里插入图片描述


基本使用

//创建ThreadLocal对象
private static ThreadLocal<String> mThreadLocal &#61; new ThreadLocal<>();static class A implements Runnable {&#64;Overridepublic void run() {//设置数据mThreadLocal.set("线程A");//覆盖数据mThreadLocal.set("线程AAA");//获取数据System.out.println(mThreadLocal.get());}
}static class B implements Runnable {&#64;Overridepublic void run() {mThreadLocal.set("线程B");System.out.println(mThreadLocal.get());}
}public static void main(String[] args) {mThreadLocal.set("线程main");new Thread(new A()).start();new Thread(new B()).start();System.out.println(mThreadLocal.get());
}

输出信息&#xff1a;

线程main
线程AAA
线程B

源码分析


Thread

public class Thread implements Runnable {ThreadLocal.ThreadLocalMap threadLocals &#61; null;private void exit() { threadLocals &#61; null;}
}

ThreadLocal

public class ThreadLocal<T> {//创建ThreadLocalMap对象void createMap(Thread t, T firstValue) {t.threadLocals &#61; new ThreadLocalMap(this, firstValue);}//获取ThreadLocalMap对象ThreadLocalMap getMap(Thread t) {return t.threadLocals;}//设置ThreadLocal变量的值//获取当前线程&#xff0c;再获取ThreadLocalMap对象&#xff0c;如果map不为null进行set操作&#xff0c;map为null则进行创建操作public void set(T value) {Thread t &#61; Thread.currentThread();ThreadLocalMap map &#61; getMap(t);if (map !&#61; null)map.set(this, value);elsecreateMap(t, value);}//获取ThreadLocal变量的值public T get() {Thread t &#61; Thread.currentThread();ThreadLocalMap map &#61; getMap(t);if (map !&#61; null) {ThreadLocalMap.Entry e &#61; map.getEntry(this);if (e !&#61; null) {T result &#61; (T)e.value;return result;}}return setInitialValue();}//初始化ThreadLocal的值private T setInitialValue() {T value &#61; initialValue();Thread t &#61; Thread.currentThread();ThreadLocalMap map &#61; getMap(t);if (map !&#61; null)map.set(this, value);else//创建createMap(t, value);return value;}//清除数据public void remove() {ThreadLocalMap m &#61; getMap(Thread.currentThread());if (m !&#61; null)m.remove(this);}private void remove(ThreadLocal<?> key) {Entry[] tab &#61; table;int len &#61; tab.length;int i &#61; key.threadLocalHashCode & (len-1);for (Entry e &#61; tab[i];e !&#61; null;e &#61; tab[i &#61; nextIndex(i, len)]) {if (e.get() &#61;&#61; key) {e.clear();expungeStaleEntry(i);return;}}}
}

ThreadLocalMap

static class ThreadLocalMap {//存储数据static class Entry extends WeakReference<ThreadLocal<?>> {Object value;Entry(ThreadLocal<?> k, Object v) {super(k);value &#61; v;}}//table初始容量private static final int INITIAL_CAPACITY &#61; 16;//table用于存储数据private Entry[] table;//下次扩容的阀值private int threshold; //2/3的负载因子private void setThreshold(int len) {threshold &#61; len * 2 / 3;}//初始化ThreadLocalMap&#xff0c;key为ThreadLocalThreadLocalMap(ThreadLocal<?> firstKey, Object firstValue) {table &#61; new Entry[INITIAL_CAPACITY];int i &#61; firstKey.threadLocalHashCode & (INITIAL_CAPACITY - 1);table[i] &#61; new Entry(firstKey, firstValue);size &#61; 1;setThreshold(INITIAL_CAPACITY);}//设置数据private void set(ThreadLocal<?> key, Object value) { Entry[] tab &#61; table;int len &#61; tab.length;//通过hash值获取索引值int i &#61; key.threadLocalHashCode & (len-1);//遍历tab如果存在则更新for (Entry e &#61; tab[i];e !&#61; null;e &#61; tab[i &#61; nextIndex(i, len)]) {ThreadLocal<?> k &#61; e.get();//key相同&#xff0c;覆盖值if (k &#61;&#61; key) {e.value &#61; value;return;}//key为null&#xff0c;清空所有key为null的数据if (k &#61;&#61; null) {replaceStaleEntry(key, value, i);return;}}//若不存在则直接添加tab[i] &#61; new Entry(key, value);int sz &#61; &#43;&#43;size;if (!cleanSomeSlots(i, sz) && sz >&#61; threshold)rehash();}//获取数据private Entry getEntry(ThreadLocal<?> key) {int i &#61; key.threadLocalHashCode & (table.length - 1);Entry e &#61; table[i];if (e !&#61; null && e.get() &#61;&#61; key)return e;elsereturn getEntryAfterMiss(key, i, e);}
}

内存泄漏问题

ThreadLocalMap内部维护了一个Entry数组&#xff0c;在没有执行remove的情况下&#xff0c;将ThreadLocal设置为null&#xff0c;下次GC时&#xff0c;key是弱引用会被回收&#xff0c;value是强引用并指向对象就会一直存在内存中&#xff0c;这样会发生内存泄露。

所以ThreadLocal使用后需要及时remove掉。


推荐阅读
  • 本文探讨了Java中有效停止线程的多种方法,包括使用标志位、中断机制及处理阻塞I/O操作等,旨在帮助开发者避免使用已废弃的危险方法,确保线程安全和程序稳定性。 ... [详细]
  • 本文提供了多个关键点来帮助开发者提高Java编程能力,包括代码规范、性能优化和最佳实践等方面,旨在指导读者成为更加优秀的Java程序员。 ... [详细]
  • 本文介绍了一个将 Java 实体对象转换为 Map 的工具类,通过反射机制获取实体类的字段并将其值映射到 Map 中,适用于需要将对象数据结构化处理的场景。 ... [详细]
  • 深入解析Java并发之ArrayBlockingQueue
    本文详细探讨了ArrayBlockingQueue,这是一种基于数组实现的阻塞队列。ArrayBlockingQueue在初始化时需要指定容量,因此它是一个有界的阻塞队列。文章不仅介绍了其基本概念和数据结构,还深入分析了其源码实现,包括各种入队、出队、获取元素和删除元素的方法。 ... [详细]
  • 本文探讨了Android系统中联系人数据库的设计,特别是AbstractContactsProvider类的作用与实现。文章提供了对源代码的详细分析,并解释了该类如何支持跨数据库操作及事务处理。源代码可从官方Android网站下载。 ... [详细]
  • 本文探讨了如何选择一个合适的序列化版本ID(serialVersionUID),包括使用生成器还是简单的整数,以及在不同情况下应如何处理序列化版本ID。 ... [详细]
  • SpringBoot底层注解用法及原理
    2.1、组件添加1、Configuration基本使用Full模式与Lite模式示例最佳实战配置类组件之间无依赖关系用Lite模式加速容器启动过程,减少判断配置类组 ... [详细]
  • 本文介绍如何通过Java代码调用阿里云短信服务API来实现短信验证码的发送功能,包括必要的依赖添加和关键代码示例。 ... [详细]
  • 在AngularJS中,有时需要在表单内包含某些控件,但又不希望这些控件导致表单变为脏状态。例如,当用户对表单进行修改后,表单的$dirty属性将变为true,触发保存对话框。然而,对于一些导航或辅助功能控件,我们可能并不希望它们触发这种行为。 ... [详细]
  • 个人博客:打开链接依赖倒置原则定义依赖倒置原则(DependenceInversionPrinciple,DIP)定义如下:Highlevelmo ... [详细]
  • 优雅地记录API调用时长
    本文旨在探讨如何高效且优雅地记录API接口的调用时长,通过实际案例和代码示例,帮助开发者理解并实施这一技术,提高系统的可观测性和调试效率。 ... [详细]
  • 深入解析C++ Atomic编程中的内存顺序
    在多线程环境中,为了防止多个线程同时修改同一数据导致的竞争条件,通常会使用内核级同步对象,如事件、互斥锁和信号量等。然而,这些方法往往伴随着高昂的上下文切换成本。本文将探讨如何利用C++11中的原子操作和内存顺序来优化多线程编程,减少不必要的开销。 ... [详细]
  • iOS 小组件开发指南
    本文详细介绍了iOS小部件(Widget)的开发流程,从环境搭建、证书配置到业务逻辑实现,提供了一系列实用的技术指导与代码示例。 ... [详细]
  • 本文探讨了一个Web工程项目的需求,即允许用户随时添加定时任务,并通过Quartz框架实现这些任务的自动化调度。文章将介绍如何设计任务表以存储任务信息和执行周期,以及如何通过一个定期扫描机制自动识别并加载新任务到调度系统中。 ... [详细]
  • 使用 ModelAttribute 实现页面数据自动填充
    本文介绍了如何利用 Spring MVC 中的 ModelAttribute 注解,在页面跳转后自动填充表单数据。主要探讨了两种实现方法及其背后的原理。 ... [详细]
author-avatar
神秘的霸气沉默_168
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有