ThreadLocal详解(附面试题),java面试常考的编程题

先自我介绍一下,小编浙江大学毕业,去过华为、字节跳动等大厂,目前阿里P7

深知大多数程序员,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年最新Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友。
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上Java开发知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新

如果你需要这些资料,可以添加V获取:vip1024b (备注Java)
img

正文

/**

  • @author xppll

  • @date 2021/12/25 21:23

*/

public class MyDemo1 {

ThreadLocal t1 = new ThreadLocal<>();

private String content;

private String getContent() {

//获取当前线程绑定的变量

return t1.get();

}

private void setContent(String content) {

// this.content = content;

//变量content绑定到当前线程

t1.set(content);

}

public static void main(String[] args) {

MyDemo1 demo = new MyDemo1();

for (int i = 0; i < 5; i++) {

Thread thread = new Thread(() -> {

demo.setContent(Thread.currentThread().getName() + “的数据”);

System.out.println(“----------------------”);

System.out.println(Thread.currentThread().getName() + “—>” + demo.getContent());

});

thread.setName(“线程” + i);

thread.start();

}

}

}

结果:

在这里插入图片描述

从结果来看,这样很好的解决了多线程之间数据隔离的问题,十分方便

2.3 ThreadLocal类与synchronized关键字

这里可能有的朋友会觉得在上述例子中我们完全可以通过加锁来实现这个功能。我们首先来看一下用synchronized代码块实现的效果:

/**

  • @author xppll

  • @date 2021/12/25 21:23

*/

public class MyDemo1 {

private String content;

private String getContent() {

return content;

}

private void setContent(String content) {

this.content = content;

}

public static void main(String[] args) {

MyDemo1 demo = new MyDemo1();

for (int i = 0; i < 5; i++) {

Thread thread = new Thread(() -> {

synchronized (MyDemo1.class) {

demo.setContent(Thread.currentThread().getName() + “的数据”);

System.out.println(“----------------------”);

System.out.println(Thread.currentThread().getName() + “—>” + demo.getContent());

}

});

thread.setName(“线程” + i);

thread.start();

}

}

}

结果:

在这里插入图片描述

从结果可以发现,加锁确实可以解决这个问题,但是在这里我们强调的是线程数据隔离的问题,并不是多线程共享数据的问题, 在这个案例中使用synchronized关键字是不合适的。而且这里使用synchronized性能会更低。使用ThreadLocal有更高的并发性。

两者区别:

| | synchronized | ThreadLocal |

| — | — | — |

| 原理 | 同步机制采用以时间换空间的方式,只提供了一份变量,让不同的线程排队访问 | ThreadLocal采用以空间换时间的方式, 为每一个线程都提供了一份变量的副本,从而实现同时访问而相不干扰 |

| 侧重点 | 多个线程之间访问资源的同步 | 多线程中让每个线程之间的数据相互隔离 |

3.ThreadLocal 的内部结构


3.1 JDK8之前的设计

如果我们不去看源代码的话,可能会猜测ThreadLocal是这样子设计的:每个ThreadLocal都创建一个Map,然后用线程作为Mapkey,要存储的局部变量作为Mapvalue,这样就能达到各个线程的局部变量隔离的效果。这是最简单的设计方法,JDK最早期的ThreadLocal 确实是这样设计的,但现在早已不是了。

在这里插入图片描述

3.2 JDK8的优化

JDK后面优化了设计方案,在JDK8中 ThreadLocal的设计是:每个Thread维护一个ThreadLocalMap,这个Map的keyThreadLocal实例本身,value才是真正要存储的值Object

具体的过程是这样的:

  • 每个Thread线程内部都有一个Map (ThreadLocalMap)

  • Map里面存储ThreadLocal对象(key)和线程的变量副本(value)

  • Thread内部的Map是由ThreadLocal维护的,由ThreadLocal负责向map获取和设置线程的变量值。

  • 对于不同的线程,每次获取副本值时,别的线程并不能获取到当前线程的副本值,形成了副本的隔离,互不干扰。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-DXzTHmEj-1641381418227)(ThreadLocal详解.assets/20210124120125266.png)]

这样设计的好处:

  1. 这样设计之后每个Map存储的Entry数量就会变少。因为之前的存储数量由Thread的数量决定,现在是由ThreadLocal的数量决定。在实际运用当中,往往ThreadLocal的数量要少于Thread的数量。

  2. 当Thread销毁之后,对应的ThreadLocalMap也会随之销毁,能减少内存的使用。

4.ThreadLocal 的核心方法源码


基于ThreadLocal的内部结构,我们继续分析它的核心方法源码,更深入的了解其操作原理。

除了构造方法之外,ThreadLocal对外暴露的方法有以下4个:、

| 方法声明 | 描述 |

| — | — |

| protected T initialValue() | 返回当前线程局部变量的初始值 |

| public void set( T value) | 设置当前线程绑定的局部变量 |

| public T get() | 获取当前线程绑定的局部变量 |

| public void remove() | 移除当前线程绑定的局部变量 |

接下来对四个方法进行源码解析

4.1 set方法

对应源码:

/**

  • 设置当前线程对应的ThreadLocal的值

  • @param value 将要保存在当前线程对应的ThreadLocal的值

*/

public void set(T value) {

// 获取当前线程对象

Thread t = Thread.currentThread();

// 获取此线程对象中维护的ThreadLocalMap对象

ThreadLocalMap map = getMap(t);

// 判断map是否存在

if (map != null)

// 存在则调用map.set设置此实体entry

map.set(this, value);

else

// 1)当前线程Thread 不存在ThreadLocalMap对象

// 2)则调用createMap进行ThreadLocalMap对象的初始化

// 3)并将 t(当前线程)和value(t对应的值)作为第一个entry存放至ThreadLocalMap中

createMap(t, value);

}

/**

  • 获取当前线程Thread对应维护的ThreadLocalMap

  • @param t the current thread 当前线程

  • @return the map 对应维护的ThreadLocalMap

*/

ThreadLocalMap getMap(Thread t) {

return t.threadLocals;

}

/**

  • 创建当前线程Thread对应维护的ThreadLocalMap

  • @param t 当前线程

  • @param firstValue 存放到map中第一个entry的值

*/

void createMap(Thread t, T firstValue) {

//这里的this是调用此方法的threadLocal

t.threadLocals = new ThreadLocalMap(this, firstValue);

}

整体可以分为几步:

  1. 首先获取当前线程,并根据当前线程获取一个Map

  2. 如果获取的Map不为空,则将参数设置到Map中(当前ThreadLocal的引用作为key)

  3. 如果Map为空,则给该线程创建 Map,并设置初始值

4.2 get方法

对应源码:

/**

  • 返回当前线程中保存ThreadLocal的值

*/

public T get() {

// 获取当前线程对象

Thread t = Thread.currentThread();

// 获取此线程对象中维护的ThreadLocalMap对象

ThreadLocalMap map = getMap(t);

// 如果此map存在

if (map != null) {

// 以当前的ThreadLocal 为 key,调用getEntry获取对应的存储实体e

ThreadLocalMap.Entry e = map.getEntry(this);

// 对e进行判空

if (e != null) {

@SuppressWarnings(“unchecked”)

// 获取存储实体 e 对应的 value值

// 即为我们想要的当前线程对应此ThreadLocal的值

T result = (T)e.value;

return result;

}

}

/*

初始化 : 有两种情况有执行当前代码

第一种情况: map不存在,表示此线程没有维护的ThreadLocalMap对象

第二种情况: map存在, 但是没有与当前ThreadLocal关联的entry

*/

return setInitialValue();

}

/**

  • 初始化

  • @return the initial value 初始化后的值

*/

private T setInitialValue() {

// 调用initialValue获取初始化的值

// 此方法可以被子类重写, 如果不重写默认返回null

T value = initialValue();

// 获取当前线程对象

Thread t = Thread.currentThread();

// 获取此线程对象中维护的ThreadLocalMap对象

ThreadLocalMap map = getMap(t);

// 判断map是否存在

if (map != null)

// 存在则调用map.set设置此实体entry

map.set(this, value);

else

// 1)当前线程Thread 不存在ThreadLocalMap对象

// 2)则调用createMap进行ThreadLocalMap对象的初始化

// 3)并将 t(当前线程)和value(t对应的值)作为第一个entry存放至ThreadLocalMap中

createMap(t, value);

// 返回设置的值value

return value;

}

整体可以分为几步:

  1. 首先获取当前线程, 根据当前线程获取一个Map

  2. 如果获取的Map不为空,则在Map中以ThreadLocal的引用作为key来在Map中获取对应的Entry e,否则转到4

  3. 如果e不为null,则返回e.value,否则转到4

  4. Map为空或者e为空,则通过initialValue函数获取初始值value,然后用ThreadLocal的引用和value作为firstKey和firstValue创建一个新的Map

总结成一句话就是:先获取当前线程的 ThreadLocalMap 变量,如果存在则返回值,不存在则创建并返回初始值。

4.3 remove方法

对应源码:

/**

  • 删除当前线程中保存的ThreadLocal对应的实体entry

*/

public void remove() {

// 获取当前线程对象中维护的ThreadLocalMap对象

ThreadLocalMap m = getMap(Thread.currentThread());

// 如果此map存在

if (m != null)

// 存在则调用map.remove

// 以当前ThreadLocal为key删除对应的实体entry

m.remove(this);

}

整体可以分为几步:

  1. 首先获取当前线程,并根据当前线程获取一个Map

  2. 如果获取的Map不为空,则移除当前ThreadLocal对象对应的entry

4.4 initialValue方法

对应源码:

/**

  • 返回当前线程对应的ThreadLocal的初始值

  • 此方法的第一次调用发生在,当线程通过get方法访问此线程的ThreadLocal值时

  • 除非线程先调用了set方法,在这种情况下,initialValue 才不会被这个线程调用。

  • 通常情况下,每个线程最多调用一次这个方法。

  • 这个方法仅仅简单的返回null {@code null};

  • 如果程序员想ThreadLocal线程局部变量有一个除null以外的初始值,

  • 必须通过子类继承{@code ThreadLocal} 的方式去重写此方法

  • 通常, 可以通过匿名内部类的方式实现

  • @return 当前ThreadLocal的初始值

*/

protected T initialValue() {

return null;

}

此方法的作用:返回该线程局部变量的初始值。

(1) 这个方法是一个延迟调用方法,从上面的代码我们得知,在set方法还未调用而先调用了get方法时才执行,并且仅执行1次

(2)这个方法缺省实现直接返回一个null

(3)如果想要一个除null之外的初始值,可以重写此方法。(备注: 该方法是一个protected的方法,显然是为了让子类覆盖而设计的)

5.ThreadLocalMap 源码分析


5.1 基本结构

ThreadLocalMapThreadLocal的内部类,没有实现Map接口,用独立的方式实现了Map的功能,其内部的Entry也是独立实现。如下图

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Z5s2krCM-1641381418228)(ThreadLocal详解.assets/20210124143602480.png)]

成员变量

/**

  • 初始容量 —— 必须是2的整次幂

*/

private static final int INITIAL_CAPACITY = 16;

/**

  • 存放数据的table,Entry类的定义在下面分析

  • 同样,数组长度必须是2的整次幂。

*/

private Entry[] table;

/**

  • 数组里面entrys的个数,可以用于判断table当前使用量是否超过阈值。

*/

private int size = 0;

/**

  • 进行扩容的阈值,表使用量大于它的时候进行扩容。

*/

private int threshold; // Default to 0

跟HashMap类似:

  • INITIAL_CAPACITY代表这个Map的初始容量

  • table 是一个Entry 类型的数组,用于存储数据

  • size 代表表中的存储数目

  • threshold 代表需要扩容时对应 size 的阈值

存储结构 - Entry

/*

  • Entry继承WeakReference,并且用ThreadLocal作为key

  • 如果key为null(entry.get() == null),意味着key不再被引用

  • 因此这时候entry也可以从table中清除

*/

static class Entry extends WeakReference<ThreadLocal<?>> {

/** The value associated with this ThreadLocal. */

Object value;

Entry(ThreadLocal<?> k, Object v) {

super(k);

value = v;

}

}

从源码就可以看出:

  • ThreadLocalMap中,也是用Entry来保存K-V结构数据的。不过Entry中的key只能是ThreadLocal对象,这点在构造方法中已经限定死了

  • 另外,Entry继承WeakReference,也就是key(ThreadLocal)是弱引用,其目的是将ThreadLocal对象的生命周期和线程生命周期解绑

5.2 弱引用和内存泄漏

先来了解一些概念:

内存相关概念

  • Memory overflow:内存溢出,没有足够的内存提供申请者使用。

  • Memory leak:内存泄漏,是指程序中已动态分配的堆内存由于某种原因程序未释放或无法释放,造成系统内存的浪费,导致程序运行速度减慢甚至系统崩溃等严重后果。内存泄漏的堆积终将导致内存溢出。

弱引用相关概念

关于Java中的引用可以参考我的这篇文章:【JVM】垃圾回收1-3节

我们接下来看几个问题,具体了解弱引用和内存泄漏:

① 我们先来看看如果key使用强引用会出现什么问题?

假设ThreadLocalMap中的key使用了强引用,此时ThreadLocal的内存图(实线表示强引用)如下:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-wdbcUx4D-1641381418230)(ThreadLocal详解.assets/20210124144330949.png)]

  1. 假设在业务代码中使用完ThreadLocal ,threadLocal Ref被回收了。

  2. 但是因为threadLocalMap的Entry强引用了threadLocal,造成threadLocal无法被回收。

  3. 在没有手动删除这个Entry以及CurrentThread依然运行的前提下,始终有强引用链 threadRef->currentThread->threadLocalMap->entry,Entry就不会被回收(Entry中包括了ThreadLocal实例和value),导致Entry内存泄漏。

  4. 也就是说,ThreadLocalMap中的key使用了强引用, 是无法完全避免内存泄漏的

② 我们在看看如果key使用弱引用呢?

ThreadLocalMap中的key使用了弱引用,此时ThreadLocal的内存图(实线表示强引用,虚线表示弱引用)如下:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-fgYTy0fl-1641381418236)(ThreadLocal详解.assets/20210124144945153.png)]

  • 同样假设在业务代码中使用完ThreadLocal ,threadLocal Ref被回收了。

  • 由于ThreadLocalMap只持有ThreadLocal的弱引用,没有任何强引用指向threadlocal实例, 所以threadlocal就可以顺利被gc回收,此时Entry中的key=null

  • 但是在没有手动删除这个Entry以及CurrentThread依然运行的前提下,也存在有强引用链 threadRef->currentThread->threadLocalMap->entry -> value ,value不会被回收, 而这块value永远不会被访问到了,导致value内存泄漏。

  • 也就是说,ThreadLocalMap中的key使用了弱引用, 也有可能内存泄漏。

③ 内存泄漏的真实原因是什么呢?

比较以上两种情况,我们就会发现,内存泄漏的发生跟ThreadLocalMap中的key是否使用弱引用是没有关系的。那么内存泄漏的的真正原因是什么呢?

如果想避免内存泄漏,都有两个前提:

  1. 只要在使用完ThreadLocal,调用其remove方法删除对应的Entry,就能避免内存泄漏。

  2. 由于ThreadLocalMap是Thread的一个属性,被当前线程所引用,所以它的生命周期跟Thread一样长。那么在使用完ThreadLocal之后,如果当前Thread也随之执行结束,ThreadLocalMap自然也会被gc回收,从根源上避免了内存泄漏。

综上,ThreadLocal内存泄漏的根源是:由于ThreadLocalMap的生命周期跟Thread一样长,如果没有手动删除对应key就会导致内存泄漏。

④ 为什么使用弱引用?

根据刚才的分析,我们知道了:无论ThreadLocalMap中的key使用哪种类型引用都无法完全避免内存泄漏,跟使用弱引用没有关系。

要避免内存泄漏有两种方式:

最后,附一张自己面试前准备的脑图:

image

面试前一定少不了刷题,为了方便大家复习,我分享一波个人整理的面试大全宝典

  • Java核心知识整理

image

  • Spring全家桶(实战系列)

image.png

Step3:刷题

既然是要面试,那么就少不了刷题,实际上春节回家后,哪儿也去不了,我自己是刷了不少面试题的,所以在面试过程中才能够做到心中有数,基本上会清楚面试过程中会问到哪些知识点,高频题又有哪些,所以刷题是面试前期准备过程中非常重要的一点。

以下是我私藏的面试题库:

image

很多人感叹“学习无用”,实际上之所以产生无用论,是因为自己想要的与自己所学的匹配不上,这也就意味着自己学得远远不够。无论是学习还是工作,都应该有主动性,所以如果拥有大厂梦,那么就要自己努力去实现它。

最后祝愿各位身体健康,顺利拿到心仪的offer!

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加V获取:vip1024b (备注Java)
img

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
使用弱引用?

根据刚才的分析,我们知道了:无论ThreadLocalMap中的key使用哪种类型引用都无法完全避免内存泄漏,跟使用弱引用没有关系。

要避免内存泄漏有两种方式:

最后,附一张自己面试前准备的脑图:

[外链图片转存中…(img-7c5Ee3lP-1713428605461)]

面试前一定少不了刷题,为了方便大家复习,我分享一波个人整理的面试大全宝典

  • Java核心知识整理

[外链图片转存中…(img-TEFkZS3U-1713428605461)]

  • Spring全家桶(实战系列)

[外链图片转存中…(img-VWh1mXNY-1713428605461)]

Step3:刷题

既然是要面试,那么就少不了刷题,实际上春节回家后,哪儿也去不了,我自己是刷了不少面试题的,所以在面试过程中才能够做到心中有数,基本上会清楚面试过程中会问到哪些知识点,高频题又有哪些,所以刷题是面试前期准备过程中非常重要的一点。

以下是我私藏的面试题库:

[外链图片转存中…(img-FeswkDGa-1713428605462)]

很多人感叹“学习无用”,实际上之所以产生无用论,是因为自己想要的与自己所学的匹配不上,这也就意味着自己学得远远不够。无论是学习还是工作,都应该有主动性,所以如果拥有大厂梦,那么就要自己努力去实现它。

最后祝愿各位身体健康,顺利拿到心仪的offer!

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加V获取:vip1024b (备注Java)
[外链图片转存中…(img-wmfVXg8t-1713428605462)]

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

  • 25
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值