首页 星云 工具 资源 星选 资讯 热门工具
:

PDF转图片 完全免费 小红书视频下载 无水印 抖音视频下载 无水印 数字星空

反问面试官3个ThreadLocal的问题

编程知识
2024年09月23日 08:20

ThreadLocal,一个Java人面试绕不开的话题,我也很奇怪为什么那些面试官很喜欢问这个,也不知道他们自己有没有搞清楚。

接下来,我想先说说ThreadLocal的用法和使用场景,然后反问面试官3个关于ThreadLocal的话题

使用方法和场景

一句话总结:ThreadLocal是给每个线程准备一份“独立的小空间”,它让每个线程都拥有自己独立的变量副本。在多个线程并发访问时,不用担心变量之间的冲突问题,避免了多线程之间的数据共享风险。

使用场景

ThreadLocal的使用场景主要在多线程环境中,能够为每个线程提供独立的变量副本。比如:

  • 用户上下文信息:比如,在Web应用中,每个请求可能由不同的线程处理,在处理用户请求时为每个线程维护独立的用户信息。
  • 数据库连接管理:比如,在多线程环境下,每个线程需要有自己独立的数据库连接。
  • 事务管理:比如,在处理事务时,每个线程可能需要有自己的事务上下文,确保线程安全的事务操作。
  • 数据传递:比如,同一个线程,在不同的方法之间传递数据,但又不想使用方法参数去传递,就可以使用ThreadLocal。像我们常用的日志跟踪场景,跟踪的ID会存在ThreadLocal中贯穿整个链条。

总之有2个场景:

  1. 在多线程场景下,每个线程需要独立管理变量的场景。
  2. 某个线程想在整条链路上共享独立变量的场景。

使用方法

使用时,记住3条核心原则:

  • 每个线程都有一份独立的数据。
  • 线程内部使用的是ThreadLocalMap来保存数据,Key就是ThreadLocal对象。
  • 使用完毕后,记得调用remove方法,防止内存溢出。

代码示例

独立保存变量的示例:

public class ThreadLocal4Independent {

    private static ThreadLocal<Integer> threadLocalVar = new ThreadLocal<>();

    public static void main(String[] args) {
        Runnable task = () -> {
            int num = (int) (Math.random() * 100);
            threadLocalVar.set(num);
            System.out.println("线程:" + Thread.currentThread().getName() + "的值:" + threadLocalVar.get());
            threadLocalVar.remove();
        };

        new Thread(task, "1").start();
        new Thread(task, "2").start();
    }

}

传递参数的示例:

public class ThreadLocal4DataPass {
    // 使用ThreadLocal来存储需要在多个方法间传递的数据
    private static final ThreadLocal<String> threadLocalData = new ThreadLocal<>();

    public static void main(String[] args) {
        // 在主线程中设置数据
        threadLocalData.set("ThreadLocal");

        // 在主线程中调用不同的方法
        method1();
        method2();
        
        // 清除ThreadLocal变量,防止内存泄露
        threadLocalData.remove();
    }

    private static void method1() {
        // 在method1中获取数据并打印
        String data = threadLocalData.get();
        System.out.println("方法1拿到的数据是:" + data);
    }

    private static void method2() {
        // 在method2中获取数据并打印
        String data = threadLocalData.get();
        System.out.println("方法2拿到的数据是:" + data);
    }
}

聊完使用场景和方法,接下来问面试官几个问题。

问题1:请画出ThreadLocal和Thread的关系图

ThreadLocal和Thread的关系图如下。

这里要牢记3点

  1. 数据实际上是存在ThreadLocalMap中的,ThreadLocalMap归Thread所持有。见源代码。

  2. ThreadLocalMap内部使用的是K-V结构,Key是我们定义的ThreadLocal对象。见源代码。

  3. ThreadLocalMap对ThreadLocal是弱引用关系。见源代码。

问题2:为什么ThreadLocalMap里的Key是弱引用

那为什么ThreadLocalMap里的Key是使用ThreadLocal呢?为什么又是弱引用呢?

这就不得不说JDK的设计者的思想非常精妙了,有3点妙处:

  1. 一个线程要是存了多种数据,总得有个规则去找他们,那就根据定义的ThreadLocal对象去找吧。
  2. 对于开发者来说,他只需要使用ThreadLocal去保存数据即可,无需关系底层结构。也就是说对外暴露简单的使用方式即可,对于不需要调用方知道的细节全部隐藏。
  3. 一般情况下Thread的生命周期会很长,比如Web容器启动后,就会启动大量的线程丢到线程池中复用。所以ThreadLocalMap的生命周期也会很长。但是,ThreadLocal对象存在的周期不一定长,如果,ThreadLocalMap的Key对ThreadLocal是强引用的话,那么ThreadLocal对象就会一直存在于内存中得不到释放,最终会导致内存溢出,所以采用了弱引用。

问题3:为什么ThreadLocal使用不当会造成内存溢出

从上图的关系图可以看出,Value的生命周期是跟着Thread的生命周期来的,如果一直不处理的话,也会出现内存溢出的情况。

为了避免内存溢出的情况,我们在使用完ThreadLocal后,要即使调用remove方法,以便JVM回收Value。

总结

ThreadLocal 是并发编程中的强大工具,能够为每个线程提供独立的变量副本,避免线程安全问题。并且这个ThreadLocal存入的值能够贯穿整个流程。使用时要注意上文的几点,防止造成内存溢出。

本篇完结!欢迎 关注、加V(yclxiao)交流、全网可搜(程序员半支烟)

原文链接:https://mp.weixin.qq.com/s/Z3x_tw1jks_KYE0ljETp0Q

From:https://www.cnblogs.com/mangod/p/18426330
本文地址: http://www.shuzixingkong.net/article/2221
0评论
提交 加载更多评论
其他文章 Redis 内存突增时,如何定量分析其内存使用情况
背景 最近碰到一个 case,一个 Redis 实例的内存突增,used_memory最大时达到了 78.9G,而该实例的maxmemory配置却只有 16G,最终导致实例中的数据被大量驱逐。 以下是问题发生时INFO MEMORY的部分输出内容。 #&#160;Memoryused_memory:
Unity中的光源类型(向前渲染路径进行光照计算)
Unity中的光源类型 Unity中共支持4种光源类型: 平行光 点光源 聚光灯 面光源(在光照烘焙时才可以发挥作用) 光源的属性: 位置 方向(到某个点的方向) 颜色 强度 衰减(到某个点的衰减) 平行光 平行光的几何定义是最简单的,平行光可以照亮的范围是无限远的,且对与场景中的各个点的方向和强度
Unity中的光源类型(向前渲染路径进行光照计算) Unity中的光源类型(向前渲染路径进行光照计算) Unity中的光源类型(向前渲染路径进行光照计算)
记一次 RabbitMQ 消费者莫名消失问题的排查
开心一刻 今天好哥们找我借钱哥们:兄弟,我最近手头紧,能不能借我点...我:我手头也不宽裕,要不你试试银行贷款或者花呗?哥们:不行,那个借了要还的我:... 问题回顾 某天下午,生产监控告警:消息积压,队列 xxx 消息数超过 100;我第一时间想到的是应用服务是不是停了,但应用服务存活监控又没有告
记一次 RabbitMQ 消费者莫名消失问题的排查 记一次 RabbitMQ 消费者莫名消失问题的排查 记一次 RabbitMQ 消费者莫名消失问题的排查
大模型应用开发初探 : 快速直观感受RAG
检索增强生成(Retrieval Augmented Generation),简称 RAG,它是根据用户输入的提示词,通过自有垂域数据库检索相关信息,然后合并成为一个新的“完备的“提示词,最后再给大模型生成较为准确的回答。这一篇,我们来快速了解下RAG(检索增强生成)并通过一个简单的DEMO来直观感
大模型应用开发初探 : 快速直观感受RAG 大模型应用开发初探 : 快速直观感受RAG 大模型应用开发初探 : 快速直观感受RAG
使用 fabric.js 开发移动端 H5 图片编辑器
移动端 H5 图片图片编辑编辑器开发实践细节,包括自定义字体、模板切换、特效字体、文字排版等功能,手把手教你开发图片编辑器。
使用 fabric.js 开发移动端 H5 图片编辑器 使用 fabric.js 开发移动端 H5 图片编辑器 使用 fabric.js 开发移动端 H5 图片编辑器
GZY.Quartz.MUI(基于Quartz的UI可视化操作组件) 2.7.0发布 新增各项优化与BUG修复
前言 时隔大半年,终于抽出空来可以更新这个组件了 (边缘化了,大概要被裁员了) 2.7.0终于发布了~ 更新内容: 1.添加API类任务的超时时间,可以通过全局配置也可以单个任务设置 2.设置定时任务日志查看默认按开始时间倒序 3.添加是否显示控制台日志的全局配置 目前支持两个参数&#160;Sho
GZY.Quartz.MUI(基于Quartz的UI可视化操作组件) 2.7.0发布 新增各项优化与BUG修复 GZY.Quartz.MUI(基于Quartz的UI可视化操作组件) 2.7.0发布 新增各项优化与BUG修复 GZY.Quartz.MUI(基于Quartz的UI可视化操作组件) 2.7.0发布 新增各项优化与BUG修复
idea运行java项目main方法报build failure错误的解决方法
本文简要介绍了当在使用 IntelliJ IDEA 运行 Java 项目的 main 方法时遇到 'Build Failure' 错误,这通常意味着在项目的构建过程中遇到了问题。解决这类问题通常需要系统地检查和调整项目设置、代码、依赖项等多个方面。
2024 年 C# 高效开发:精选实用类库
前言 在平时开发中,好的类库能帮助我们快速实现功能,C#有很多封装好的类库。 本文将介绍一些2024年特别受欢迎的C#类库,并分析各自的优点让我们编程写代码变的更轻松、更快捷。 快来看一看有没有大家常用的类库,欢迎各位小伙伴留言补充。 1、Entity Framework Core Entity F
2024 年 C# 高效开发:精选实用类库