>>分享Java编程技术,对《Java面向对象编程》等书籍提供技术支持 书籍支持  卫琴直播  品书摘要  在线测试  资源下载  联系我们
发表一个新主题 开启一个新投票 回复文章 您是本文章第 26690 个阅读者 刷新本主题
 * 贴子主题:  终于明白阿里百度这样的大公司,为什么面试经常拿ThreadLocal考验求职者了 回复文章 点赞(0)  收藏  
作者:sunshine    发表时间:2019-08-16 17:33:16     消息  查看  搜索  好友  邮件  复制  引用

   什么是ThreadLocal

  ThreadLocal是一个本地线程副本变量工具类,各个线程都拥有一份线程私有的数据,线程之间的变量互不干扰,在高并发场景下,可以实现无状态的调用。

           ThreadLocal提供了线程安全的另一种思路,我们平常说的线程安全主要是保证共享数据的并发访问问题,通过sychronized锁或者CAS无锁策略来保证数据的一致性。    

   ThreadLocal结构图

      点击在新窗口中浏览原图
CTRL+鼠标滚轮放大或缩小

       从上面的结构图,我们已经窥见ThreadLocal的核心机制:
  •    每个Thread线程内部都有一个Map。
  •    Map里面存储线程本地对象(key)和线程的变量副本(value)
  •    Thread内部的Map是由ThreadLocal维护的,由ThreadLocal负责向map获取和设置线程的变量值。
  对于不同的线程,每次获取副本值时,别的线程并不能获取到当前线程的副本值,形成了副本的隔离,彼此之间互不干扰。

  我们来看个例子。

  下面的例子有3个线程,,修改类变量initValue,当类变量是ThreadLocal的时候3个线程修改的值互不影响,打印的结果都是66

          点击在新窗口中浏览原图
CTRL+鼠标滚轮放大或缩小

   上面的例子3个线程是如果做到同时独立修改变量的,答案就在ThreadLocal的set(),get()方法里面.

    下面我们再来看看ThreadLocal

  ThreadLocal类提供如下几个核心方法:

点击在新窗口中浏览原图
CTRL+鼠标滚轮放大或缩小    
  •    get()方法用于获取当前线程的副本变量值。
  •    set()方法用于保存当前线程的副本变量值。
  •    initialValue()为当前线程初始副本变量值。
  •    remove()方法移除当前前程的副本变量值。

     get()方法

点击在新窗口中浏览原图
CTRL+鼠标滚轮放大或缩小    
  1.    获取当前线程的ThreadLocalMap对象threadLocals
  2.    从map中获取线程存储的K-V Entry节点。
  3.    从Entry节点获取存储的Value副本值返回。
  4.    map为空的话返回初始值null,即线程变量副本为null,需要注意的是在使用中要判断是否为空指针NullPointerException。

     set()方法

      点击在新窗口中浏览原图
CTRL+鼠标滚轮放大或缩小    
  1.    获取当前线程的成员变量map
  2.    map非空,则重新将ThreadLocal和新的value副本放入到map中。
  3.    map空,则对线程的成员变量ThreadLocalMap进行初始化创建,并将ThreadLocal和value副本放入map中。

     remove()方法

点击在新窗口中浏览原图
CTRL+鼠标滚轮放大或缩小

      Thread线程内部的Map在类中描述如下:

          点击在新窗口中浏览原图
CTRL+鼠标滚轮放大或缩小

      可以看到,这个ThreadLocalMap是线程中的变量,也就是说每个线程都是相互独立的

点击在新窗口中浏览原图
CTRL+鼠标滚轮放大或缩小    

     应用场景

  类似单例类TransactionSynchronizationManager,

  RequestContextHolder中就是通过ThreadLocal保存各自线程变量的副本,这样就不需要重新创建类。

      一个知识点延伸出这么多知识点,关于弱引用、 内存优化等,不仅能考验求职者的对该知识点的掌握程度,又能考验求职者的知识面,难怪阿里百度这样的大公司喜欢在面试时拿它来考验求职者。

----------------------------
原文链接:https://blog.csdn.net/X8i0Bev/article/details/101086604

程序猿的技术大观园:www.javathinker.net



[这个贴子最后由 flybird 在 2020-01-11 20:10:30 重新编辑]
  Java面向对象编程-->输入与输出(下)
  JavaWeb开发-->访问数据库(Ⅰ)
  JSP与Hibernate开发-->JPA API的高级用法
  Java网络编程-->Socket用法详解
  精通Spring-->通过Vuex进行状态管理
  Vue3开发-->通过Vuex进行状态管理
  [原创]汽車美容
  解决javac编译错误: 编码UTF8/GBK的不可映射字符
  Java设计模式: 单一职责原则和依赖倒置原则详解
  面向对象的七大原则,你知道几个?
  使用策略模式优化代码实践,如何让项目快速起飞
  靠一个HashMap的讲解打动了头条面试官
  正则表达式范例
  正则表达式性能调优
  Java入门实用代码:线程状态监测
  Java 入门实用代码:汉诺塔算法
  Java入门实用代码:字符串替换
  Java程序初始化顺序(一看就懂)
  java Pattern和Matcher详解
  Java性能优化总结
  Java中的main()方法详解
  更多...
 IPIP: 已设置保密
楼主      
1页 0条记录 当前第1
发表一个新主题 开启一个新投票 回复文章


中文版权所有: JavaThinker技术网站 Copyright 2016-2026 沪ICP备16029593号-2
荟萃Java程序员智慧的结晶,分享交流Java前沿技术。  联系我们
如有技术文章涉及侵权,请与本站管理员联系。