优化Java多线程应用:是创建Thread对象直接调用start()方法?还是用个变量调用?

问题背景

偶然间发现一个有意思的点,平时写代码的时候,下面这两种写法,不知道大家经常使用的是哪一种写法?你能发现他俩有啥区别吗?

Java 复制代码
/** 写法一 **/
Thread thread = new Thread(() -> {
    // 线程执行的任务
    // ...
});
thread.start(); // 启动线程

/** 写法二 **/
new Thread(() -> {
    // 线程执行的任务
    // ...
}).start(); 

结果分析

建议直接调用Thread对象的start()方法而不是保持对线程对象的强引用,这一点在使用ThreadLocal时尤为重要。ThreadLocal为每个线程提供了线程局部变量的存储,这些变量是线程隔离的,并且通常用于避免多线程间的共享状态和同步问题。下面是一些关键点和代码示例,说明为什么在使用ThreadLocal时应该避免对Thread对象保持强引用:

1、内存泄漏风险

ThreadLocal使用Thread对象的ThreadLocalMap来存储线程局部变量。如果Thread对象由于外部强引用而没有被垃圾回收器回收,那么ThreadLocalMap中的条目也会保持在内存中,导致内存泄漏。

ini 复制代码
Thread thread = new Thread(() -> {
    // 使用ThreadLocal存储数据
    ThreadLocal<String> threadLocal = ThreadLocal.withInitial(() -> "ThreadLocal Value");
    // ...
});
thread.start(); // 正确做法:启动线程后不保持对Thread对象的强引用
// thread = null; // 推荐做法:线程启动后释放对Thread对象的引用

2、线程生命周期管理

当线程执行完毕后,如果它没有被外部强引用,垃圾回收器可以回收Thread对象和相关的资源。保持对Thread对象的强引用可能会导致线程资源长时间不被释放,尤其是当线程长时间运行或处于等待状态时。

java 复制代码
Thread thread = new Thread(() -> {
    // 线程执行的任务
    // ...
});
thread.start(); // 启动线程
// 无需保持强引用,线程将自行结束

3、避免不必要的线程控制

保持对Thread对象的强引用可能会诱使程序员进行不必要的线程控制,如尝试中断线程或等待线程结束。这不仅增加了代码复杂性,而且可能会干扰线程的自然生命周期。

java 复制代码
Thread thread = new Thread(() -> {
    // 线程执行的任务
    // ...
});
thread.start(); // 启动线程
// 不需要等待线程结束,除非有特定的理由
// thread.join(); // 仅在确实需要等待线程结束时使用

4、简化代码逻辑

直接启动线程并让线程自行结束,可以使代码更加简洁和易于理解。这种做法减少了管理线程生命周期的复杂性,有助于编写清晰和可维护的代码。

java 复制代码
new Thread(() -> {
    // 线程执行的任务
    // ...
}).start(); // 启动线程,无需手动管理线程生命周期

直接调用Thread对象的start()方法并避免保持对其的强引用,有助于防止内存泄漏,简化线程生命周期的管理,避免不必要的线程控制,并使代码逻辑更加清晰和简洁。这是在使用ThreadLocal和线程时的推荐做法。

相关推荐
HelloWord~1 小时前
SpringSecurity+vue通用权限系统2
java·vue.js
让我上个超影吧1 小时前
黑马点评【基于redis实现共享session登录】
java·redis
00后程序员1 小时前
提升移动端网页调试效率:WebDebugX 与常见工具组合实践
后端
HyggeBest1 小时前
Mysql的数据存储结构
后端·架构
TobyMint2 小时前
golang 实现雪花算法
后端
G探险者2 小时前
【案例解析】一次 TIME_WAIT 导致 TPS 断崖式下降的排查与优化
后端
BillKu2 小时前
Java + Spring Boot + Mybatis 插入数据后,获取自增 id 的方法
java·tomcat·mybatis
全栈凯哥2 小时前
Java详解LeetCode 热题 100(26):LeetCode 142. 环形链表 II(Linked List Cycle II)详解
java·算法·leetcode·链表
chxii2 小时前
12.7Swing控件6 JList
java
全栈凯哥2 小时前
Java详解LeetCode 热题 100(27):LeetCode 21. 合并两个有序链表(Merge Two Sorted Lists)详解
java·算法·leetcode·链表