Threadlocal相关问题
创始人
2025-05-30 04:51:51

Threadlocal相关问题

核心问题

​ 1、 什么是ThreadLocal?核心方法有哪些?

​ 2、使用 拦截器 + ThreadLocal 处理鉴权的整体流程是什么样的?

​ 3、代码编写实现流程有哪些?

面试相关

ThreadLocal内存泄漏问题

​ 1)、ThreadLocal的实现原理

​ 2)、内存泄漏原因

​ 3)、为什么ThreadLocalMap中的key使用弱引用而不是强引用?

​ 4)、正确使用姿势

问题解析

1、 什么是ThreadLocal?核心方法有哪些?

​ 原理:每个线程有一个与之绑定的容器:Map,这个Map中保存着当前线程的所有的本地变量,这个Map

​ 叫ThreadLocalMap,这个map的键为当前ThreadLocal对象,值为放进去的值

  • 线程内部的存储类,赋予了线程存储数据的能力。

  • 线程内调用get方法都可以从ThreadLocal中获取同一个对象。

  • 多个线程中ThreadLocal数据相互隔离

  • threadlocal核心方法有三个:set(存储数据)、get(获取数据)、remove(删除数据,避免内存溢出)

    ThreadLocal threadLocal = new ThreadLocal();
    threadLocal.set() //将数据绑定到当前线程
    threadLocal.get() //从当前线程中获取数据

    threadLocal.remove() //从当前线程删除数据

2、使用 拦截器 + ThreadLocal 处理鉴权的整体流程是什么样的?

在这里插入图片描述

在这里插入图片描述

3、代码编写实现流程有哪些?

​ 1)、编写ThreadLocal工具类,提供线程存储和获取对象的方法

​ 2)、拦截器解析token验证通过后,将对象存入Threadlocal中

​ 3)、修改各个Controller方法,从ThreadLocal中获取当前登录用户对象

​ 4)、拦截器的afterCompletion中调用remove方法进行threadlocal数据清空,避免内存溢出

拓展:ThreadLocal内存泄漏问题

1、ThreadLocal的实现原理

​ 每一个Thread维护一个ThreadLocalMap,key为使用弱引用的ThreadLocal实例,value为线程变量的副本。这些对象之间的引用关系如下(实心箭头表示强引用,空心箭头表示弱引用):

在这里插入图片描述

2、内存泄漏原因:

ThreadLocalMap使用ThreadLocal的弱引用作为key,如果一个ThreadLocal不存在外部强引用时,Key(ThreadLocal)势必会被GC回收,这样就会导致ThreadLocalMap中key为null, 而value还存在着强引用,只有thead线程退出以后,value的强引用链条才会断掉。

但如果当前线程再迟迟不结束的话,这些key为null的Entry的value就会一直存在一条强引用链:

Thread Ref -> Thread -> ThreaLocalMap -> Entry -> value

这就就会永远无法回收,造成内存泄漏。

3、为什么ThreadLocalMap中的key使用弱引用而不是强引用?

如果key 使用强引用:当ThreadLocalMap的key为强引用回收ThreadLocal时,因为ThreadLocalMap还持有ThreadLocal的强引用,如果没有手动删除,ThreadLocal不会被回收,导致Entry内存泄漏。

如果key使用弱引用:当ThreadLocalMap的key为弱引用回收ThreadLocal时,由于ThreadLocalMap持有ThreadLocal的弱引用,即使没有手动删除,ThreadLocal也会被回收。当key为null,在下一次ThreadLocalMap调用set(),get(),remove()方法的时候会被清除value值。使用弱引用可以让GC回收方面多一层保障:弱引用ThreadLocal不会内存泄漏,对应的value会在下一次ThreadLocalMap调用set(),get(),remove()的时候会被清除。因此,ThreadLocal内存泄漏的根源是:由于ThreadLocalMap的生命周期跟Thread一样长,如果没有手动删除对应key就会导致内存泄漏,而不是因为弱引用

但是,当key为null后,之后再也没有调用过set()、get()、remove()方法,这样就会使value一直不会被回收

4、正确使用姿势:

  • 每次使用完ThreadLocal都调用它的remove()方法清除数据
  • 将ThreadLocal变量定义成private static,这样就一直存在ThreadLocal的强引用,也就能保证任何时候都能通过ThreadLocal的弱引用访问到Entry的value值,进而清除掉 。

相关内容

热门资讯

Python基础(十七):装饰... 闭包闭包(英语:Closure),又称词法闭...
计算机科学导论笔记(十四) 目录 十六、安全 16.1 引言 16.1.1 安全目标 16.1.2 攻击 16.1.2.1 威...
@Transactional导... 首先我有一个Class A和Class B,A和B存在循环依赖。 @Servi...
HTML5-表单 HTML5-表单 一、Form 1.action 属性 action 属性用于指定表单...
【小猫爪】AUTOSAR学习笔... 【小猫爪】AUTOSAR学习笔记05-Communication Stack之CanSM模块前言1 ...
c# 使用AutoResetE...         做项目时有一个需求。用一个线程去执行耗时操作。另一个线程需要使用第一个线程的操作结果...
在pycharm中使用chat... 目录 前言 一、插件安装 二、使用步骤 总结 前言 ChatGPT是目前最强大的AI,...
Codeforces Roun... G. Subsequence Addition 标签 规律、数学 链接 传送门、 结论 当前前缀和小...
算法leetcode|42. ... 文章目录42. 接雨水:样例 1:样例 2:提示ÿ...
【项目设计】负载均衡在线OJ 🎇Linux: 博客主页:一起去看日落吗分享博主的在L...
Java开发 | 重写 | 多... 前言 大家好,我是程序猿爱打拳,今天给大家带来的是面向对象之封装继承多...
【Unity】NavMesh ... 在Unity中,可以使用自带导航系统(Navigation System...
由文心一言发布会引发的思考,聊... 文章目录前言一. 文心一言的试用1.1 文心一言发布会1.2 文心一言图片生成功能试用1.3 文心一...
java线程之Thread类的... Thread类的基本用法1. Thread类的构造方法2. Thread的几个常见属性常见属性线程中...
css实现3D弹性按钮以及bo... box-shadow 在实现案例之前先了解css的阴影属性box-shadow,该属性...
【Linux】基础命令大全、实... 个人简介:Java领域新星创作者;阿里云技术博主、星级博主、专家博主&#...
R语言基础教程4:列表和数据框 文章目录列表数据帧表头 R语言系列:1 编程基础💎2 循环语句...
Git基础知识 Git基础知识前言一、Git基本概念1、分布式版本控制系统--Git2、Git配置命令3、Git原理...
【JavaWeb】MySQL 一、数据库的相关概念 1.数据库(DB) ==存储和管...
CPU 是如何执行程序的 代码写了那么多,你知道 a = 1 + 2 这条代码是怎么被 CPU ...
从产品的角度看缓存 文章目录 1. What——什么是缓存?2. Why——为什么需要使用缓存?2.1 什么是用户体验2...
vivado 开发过程中所遇错...  [Synth 8-4556] 开辟的数组内存空间大小问题 [Synth 8-4556] size...
1.4 K8S入门之POD和网... POD 分类 自主式POD控制器管理的POD 容器 每个容器独立存在,有自己的IP地址...
【二】一起算法---队列:ST... 纸上得来终觉浅,绝知此事要躬行。大家好!我是霜淮子,欢迎订...
在使用fastjson中遇到的... 一、在使用fastjson中遇到的问题 导论:最近在写一个JavaFx项目的时候使用...
HJ31 单词倒排 描述 对字符串中的所有单词进行倒排。 说明: 1、构成单词的字符只有26个大写或小写英...
普通插槽、具名插槽、作用域插槽 插槽 插槽就是子组件提供给父组件的占位符,用slot来表示,父组件可以在...
Go语言必知必会100问题-0... 减少代码的嵌套层数 软件开发中的“心智模型”用于描述开发人员在编码时心理活动,每段代码...
CSRF漏洞的概念、利用方式、... CSRF漏洞1.CSRF的概念1.1 什么是CSRF?1.2 基本攻击流程2.CSRF...
基于springboot开发的... 基于springboot开发的学生考勤管理系统 如需更多资料请前往文章底部获取联系方式 系统设计主要...