【高并发】Java 多线程与线程池实战:高性能并发编程指南

Scroll Down

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. 线程池使用建议

  1. 避免使用 Executors 创建线程池(默认参数可能导致 OOM),推荐 ThreadPoolExecutor
  2. 核心线程数 = CPU 核心数Runtime.getRuntime().availableProcessors())。
  3. I/O 密集型任务 可用 newCachedThreadPool(),减少等待时间。
  4. CPU 密集型任务 推荐 固定线程池newFixedThreadPool())。
  5. 长时间不使用线程池时,记得 shutdown() 释放资源

总结

多线程能提升程序效率,适用于高并发任务
线程池能避免资源浪费,提高系统吞吐量
推荐使用 ThreadPoolExecutor 进行精细化线程管理

希望这篇文章对你有所帮助!欢迎关注,持续输出高质量 Java 并发编程内容 🚀🚀🚀