Java 多线程与线程池实战:高性能并发编程指南
在 Java 开发中,多线程编程是提升性能的关键技术之一。合理使用线程可以提高系统的并发能力,而使用 线程池(ThreadPoolExecutor) 能有效地管理和优化线程资源,避免频繁创建和销毁线程带来的开销。本文将深入解析 Java 多线程与线程池的核心概念,并提供实战代码示例,助你掌握高效并发编程。
1. 为什么需要多线程?
在单线程程序中,所有任务都 顺序执行,如果某个任务特别耗时,整个程序都会被阻塞。例如,以下代码模拟了一个 读取数据库+计算+写入文件 的业务场景:
public class SingleThreadDemo {
public static void main(String[] args) {
long start = System.currentTimeMillis();
readFromDatabase(); // 读取数据库(假设耗时 2s)
processData(); // 计算数据(假设耗时 3s)
writeToFile(); // 写入文件(假设耗时 2s)
System.out.println("总耗时: " + (System.currentTimeMillis() - start) + " ms");
}
static void readFromDatabase() { try { Thread.sleep(2000); } catch (InterruptedException e) {} }
static void processData() { try { Thread.sleep(3000); } catch (InterruptedException e) {} }
static void writeToFile() { try { Thread.sleep(2000); } catch (InterruptedException e) {} }
}
执行时间:
总耗时: 7000 ms
问题分析:
- 所有任务顺序执行,总耗时 = 2s + 3s + 2s = 7s。
- 由于单线程执行,I/O 任务(数据库和文件操作)会阻塞 CPU,导致低效。
解决方案:使用多线程!
2. Java 多线程的几种实现方式
(1)继承 Thread
类
class MyThread extends Thread {
public void run() {
System.out.println("线程 " + Thread.currentThread().getName() + " 正在执行");
}
}
public class ThreadExample {
public static void main(String[] args) {
MyThread t1 = new MyThread();
MyThread t2 = new MyThread();
t1.start();
t2.start();
}
}
缺点:
- Java 不支持多继承,如果
MyThread
需要继承其他类,则无法继承Thread
。 - 线程资源不可共享,多个
Thread
实例会创建多个线程,影响性能。
(2)实现 Runnable
接口
class MyRunnable implements Runnable {
public void run() {
System.out.println("线程 " + Thread.currentThread().getName() + " 正在执行");
}
}
public class RunnableExample {
public static void main(String[] args) {
Thread t1 = new Thread(new MyRunnable());
Thread t2 = new Thread(new MyRunnable());
t1.start();
t2.start();
}
}
优点:
✅ Runnable
适用于资源共享,多个线程可以共享同一个 Runnable
实例。
✅ Runnable
不影响继承结构,可以继续继承其他类。
(3)实现 Callable
(支持返回值)
Runnable
不能返回结果,而 Callable
可以返回计算结果:
import java.util.concurrent.*;
class MyCallable implements Callable<Integer> {
public Integer call() {
return 42;
}
}
public class CallableExample {
public static void main(String[] args) throws Exception {
ExecutorService executor = Executors.newSingleThreadExecutor();
Future<Integer> future = executor.submit(new MyCallable());
System.out.println("计算结果: " + future.get()); // 阻塞等待结果
executor.shutdown();
}
}
✅ 适用于需要返回计算结果的任务(如数据处理、AI 计算等)。
3. 线程池(ThreadPoolExecutor)详解
为什么要使用线程池?
- 频繁创建/销毁线程消耗资源,线程池能复用线程,减少开销。
- 控制并发数量,防止过多线程导致内存溢出。
- 任务调度,更好地管理任务队列。
常见线程池
线程池类型 | 说明 |
---|---|
newFixedThreadPool(n) |
固定大小线程池,适用于任务量稳定的场景 |
newCachedThreadPool() |
可扩展线程池,适用于大量短任务 |
newSingleThreadExecutor() |
单线程池,适用于顺序执行任务 |
newScheduledThreadPool(n) |
定时任务线程池 |
(1)固定线程池 FixedThreadPool
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
public class FixedThreadPoolExample {
public static void main(String[] args) {
ExecutorService executor = Executors.newFixedThreadPool(3);
for (int i = 1; i <= 5; i++) {
int taskId = i;
executor.execute(() -> System.out.println("任务 " + taskId + " 执行于 " + Thread.currentThread().getName()));
}
executor.shutdown();
}
}
✅ 适用于任务量稳定的高并发场景(如 Web 服务器处理请求)。
(2)自定义 ThreadPoolExecutor
可以自由调整线程数、任务队列和拒绝策略:
import java.util.concurrent.*;
public class CustomThreadPool {
public static void main(String[] args) {
ThreadPoolExecutor executor = new ThreadPoolExecutor(
2, // 核心线程数
4, // 最大线程数
60, TimeUnit.SECONDS, // 线程空闲时间
new LinkedBlockingQueue<>(2), // 任务队列(2个)
Executors.defaultThreadFactory(),
new ThreadPoolExecutor.CallerRunsPolicy() // 拒绝策略(让主线程执行)
);
for (int i = 1; i <= 6; i++) {
int taskId = i;
executor.execute(() -> System.out.println("任务 " + taskId + " 执行于 " + Thread.currentThread().getName()));
}
executor.shutdown();
}
}
✅ 适用于高并发场景,可自由调整参数!
4. 线程池使用建议
- 避免使用
Executors
创建线程池(默认参数可能导致 OOM),推荐ThreadPoolExecutor
。 - 核心线程数 = CPU 核心数(
Runtime.getRuntime().availableProcessors()
)。 - I/O 密集型任务 可用
newCachedThreadPool()
,减少等待时间。 - CPU 密集型任务 推荐 固定线程池(
newFixedThreadPool()
)。 - 长时间不使用线程池时,记得
shutdown()
释放资源。
总结
✅ 多线程能提升程序效率,适用于高并发任务。
✅ 线程池能避免资源浪费,提高系统吞吐量。
✅ 推荐使用 ThreadPoolExecutor
进行精细化线程管理。
希望这篇文章对你有所帮助!欢迎关注,持续输出高质量 Java 并发编程内容 🚀🚀🚀