文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

反问面试官三个 ThreadLocal 的问题

2024-11-29 19:13

关注

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

使用方法和场景

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

1.使用场景

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

总之有2个场景:

2.使用方法

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

代码示例

独立保存变量的示例:

public class ThreadLocal4Independent {

    private static ThreadLocal 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 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点妙处:

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

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

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

总结

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

来源:程序员半支烟内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯