ThreadLocal底层原理和实现Demo
- 每天一个面试题:ThreadLocal
- 实现ThreadLocal的Demo
- ThreadLocal底层原理
- 为什么ThreadLocalMap的key设计为弱引用
今天开始一个新专栏:每天一个面试题系列
也没有拿到令人心动的offer,看来自己学习方式和能力还是差很多,痛定思痛,改变自己的学习方式,辞去原有的后端实习,
开始全新的学习,沉淀才会有产出,一步一脚印!
面试题系列搞起来,这个专栏并非单纯的八股文,我会在技术底层的基础上,不至于Debug,还会做一些实例的实现,实现一些简单的Demo,或者用于我做过的项目中去;
代码会同步在我的gitee中去,觉得不错的同学记得一键三连求关注,感谢:
链接: ThreadLocal
每天一个面试题:ThreadLocal
-
ThreadLocal表示线程的“局部变量”,它确保每个线程的ThreadLocal变量都是各自独立的;
-
ThreadLocal适合在一个线程的处理流程中保持上下文(避免了同一参数在所有方法中传递);
-
使用ThreadLocal要用try … finally结构,并在finally中清除。
实现ThreadLocal的Demo
多个线程,我们希望每个线程内只连接到自己的数据库
public class DemoThreadLocal {
public static void main(String[] args) {
test();
}
private static void test() {
for (int i = 0; i < 5; i++) {
new Thread(()->{
//多个线程,我们希望每个线程内只连接到自己的数据库,这里的打印相当 各个用户 操作自己的数据库
System.out.println(Utils.getConnection().toString());
System.out.println(Utils.getConnection().toString());
System.out.println(Utils.getConnection().toString());
}, "i" + (i+1)).start();
}
}
static class Utils{
private static final ThreadLocal<Connection> threadLocal = new ThreadLocal<Connection>();
// 通过本地线程管理各自的数据库连接
public static Connection getConnection(){
Connection con = threadLocal.get();
if (con==null) {
con = getIn();
threadLocal.set(con);
}
return con;
}
//配置数据库连接
public static Connection getIn(){
try {
return DriverManager.getConnection("jdbc:mysql://localhost:3300/blog?useSSL=false","root","root");
}catch (Exception e){
throw new RuntimeException(e);
}
}
}
}
ThreadLocal底层原理
通过hash计算去计算资源对应的位置索引,进行存放
如果出现了Hash冲突:拉链法、其他算法
为什么ThreadLocalMap的key设计为弱引用
强引用无法被GC,所以使用ThreadLocal要用try … finally结构,并在finally中清除。
ThreadMap在get时,如果没有值,也会执行set (key,null)操作
三大情况
- 发现key= null时
- GC自动收回
- 手动remove