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

简介: 这篇文章探讨了Java中两种创建和启动线程的方法,并分析了它们的区别。作者建议直接调用 `Thread` 对象的 `start()` 方法,而非保持强引用,以避免内存泄漏、简化线程生命周期管理,并减少不必要的线程控制。文章详细解释了这种方法在使用 `ThreadLocal` 时的优势,并提供了代码示例。作者洛小豆,文章来源于稀土掘金。

问题背景

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

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和线程时的推荐做法。



转载来源:https://juejin.cn/post/7359461815376986163

相关文章
|
2天前
|
存储 缓存 安全
【Java面试题汇总】多线程、JUC、锁篇(2023版)
线程和进程的区别、CAS的ABA问题、AQS、哪些地方使用了CAS、怎么保证线程安全、线程同步方式、synchronized的用法及原理、Lock、volatile、线程的六个状态、ThreadLocal、线程通信方式、创建方式、两种创建线程池的方法、线程池设置合适的线程数、线程安全的集合?ConcurrentHashMap、JUC
【Java面试题汇总】多线程、JUC、锁篇(2023版)
|
4天前
|
Java
Java的方法详解
Java的方法是类中的重要组成部分,用于定义类的行为。方法可以接收参数、执行操作并返回结果。其基本语法包括返回类型、方法名、参数列表和方法体。方法支持重载,即同名但参数不同的多个方法;静态方法则直接通过类名调用,无需实例化。此外,Java还支持可变参数,允许方法接收不定数量的参数。通过访问修饰符如`public`、`protected`、`private`,可以控制方法的可见性。方法是实现类功能的基本单元,增强了程序的灵活性和复用性。
|
2天前
|
Java 调度 开发者
Java中的多线程基础及其应用
【9月更文挑战第13天】本文将深入探讨Java中的多线程概念,从基本理论到实际应用,带你一步步了解如何有效使用多线程来提升程序的性能。我们将通过实际代码示例,展示如何在Java中创建和管理线程,以及如何利用线程池优化资源管理。无论你是初学者还是有经验的开发者,这篇文章都将为你提供有价值的见解和技巧,帮助你更好地理解和应用多线程编程。
|
2天前
|
JavaScript 前端开发 Java
通过JUnit5访问Java静态、私有、保护变量和方法
在《通过Gtest访问C++静态、私有、保护变量和方法》一文中介绍了如何通过Gtest访问C++静态、私有、保护变量和方法,本文介绍如何通过Junit5访问Java静态、私有、保护变量和方法。
8 0
|
20天前
|
存储 监控 Java
Java多线程优化:提高线程池性能的技巧与实践
Java多线程优化:提高线程池性能的技巧与实践
46 1
|
3天前
|
Java 数据库 Android开发
一个Android App最少有几个线程?实现多线程的方式有哪些?
本文介绍了Android多线程编程的重要性及其实现方法,涵盖了基本概念、常见线程类型(如主线程、工作线程)以及多种多线程实现方式(如`Thread`、`HandlerThread`、`Executors`、Kotlin协程等)。通过合理的多线程管理,可大幅提升应用性能和用户体验。
24 15
一个Android App最少有几个线程?实现多线程的方式有哪些?
|
5天前
|
Java 数据库 Android开发
一个Android App最少有几个线程?实现多线程的方式有哪些?
本文介绍了Android应用开发中的多线程编程,涵盖基本概念、常见实现方式及最佳实践。主要内容包括主线程与工作线程的作用、多线程的多种实现方法(如 `Thread`、`HandlerThread`、`Executors` 和 Kotlin 协程),以及如何避免内存泄漏和合理使用线程池。通过有效的多线程管理,可以显著提升应用性能和用户体验。
23 10
|
12天前
|
存储 Ubuntu Linux
C语言 多线程编程(1) 初识线程和条件变量
本文档详细介绍了多线程的概念、相关命令及线程的操作方法。首先解释了线程的定义及其与进程的关系,接着对比了线程与进程的区别。随后介绍了如何在 Linux 系统中使用 `pidstat`、`top` 和 `ps` 命令查看线程信息。文档还探讨了多进程和多线程模式各自的优缺点及适用场景,并详细讲解了如何使用 POSIX 线程库创建、退出、等待和取消线程。此外,还介绍了线程分离的概念和方法,并提供了多个示例代码帮助理解。最后,深入探讨了线程间的通讯机制、互斥锁和条件变量的使用,通过具体示例展示了如何实现生产者与消费者的同步模型。
|
20天前
|
监控 安全 Java
Java多线程调试技巧:如何定位和解决线程安全问题
Java多线程调试技巧:如何定位和解决线程安全问题
68 2