java多线程打印

模块一:线程交替打印(奇数 / 偶数,1~10)

synchronized + wait()/notify() 实现

java 复制代码
/**
 * 线程交替打印 1~10(奇数线程 + 偶数线程)
 * 核心:synchronized 加锁 + wait()/notify() 线程通信,实现交替执行
 */
public class AlternatePrintDemo {
    // 1. 共享计数器(控制打印数字,从1开始)
    private static int count = 1;
    // 2. 共享锁对象(保证线程安全,作为线程通信的载体)
    private static final Object LOCK = new Object();

    public static void main(String[] args) {
        // 线程1:打印奇数(1、3、5、7、9)
        Thread oddThread = new Thread(() -> printOdd(), "奇数线程");
        // 线程2:打印偶数(2、4、6、8、10)
        Thread evenThread = new Thread(() -> printEven(), "偶数线程");

        // 启动线程(启动顺序不影响结果)
        oddThread.start();
        evenThread.start();
    }

    /**
     * 打印奇数逻辑
     */
    private static void printOdd() {
        while (count <= 10) {
            synchronized (LOCK) { // 加锁:获取LOCK对象的监视器锁
                try {
                    // 不是奇数,进入等待(释放锁,等待偶数线程唤醒)
                    if (count % 2 == 0) {
                        LOCK.wait();
                    }
                    // 打印奇数并自增
                    if (count <= 10) {
                        System.out.println(Thread.currentThread().getName() + ":" + count);
                        count++;
                    }
                    // 唤醒等待在LOCK上的另一个线程(偶数线程)
                    LOCK.notify();
                } catch (InterruptedException e) {
                    // 中断异常处理:恢复线程中断状态,符合最佳实践
                    Thread.currentThread().interrupt();
                }
            }
        }
    }

    /**
     * 打印偶数逻辑
     */
    private static void printEven() {
        while (count <= 10) {
            synchronized (LOCK) { // 加锁:获取LOCK对象的监视器锁
                try {
                    // 不是偶数,进入等待(释放锁,等待奇数线程唤醒)
                    if (count % 2 == 1) {
                        LOCK.wait();
                    }
                    // 打印偶数并自增
                    if (count <= 10) {
                        System.out.println(Thread.currentThread().getName() + ":" + count);
                        count++;
                    }
                    // 唤醒等待在LOCK上的另一个线程(奇数线程)
                    LOCK.notify();
                } catch (InterruptedException e) {
                    // 中断异常处理:恢复线程中断状态,符合最佳实践
                    Thread.currentThread().interrupt();
                }
            }
        }
    }
}

打印结果:

java 复制代码
奇数线程:1
偶数线程:2
奇数线程:3
偶数线程:4
奇数线程:5
偶数线程:6
奇数线程:7
偶数线程:8
奇数线程:9
偶数线程:10

模块二:线程等待打印(线程 1 先执行,线程 2 后执行)

方案 A:Thread.join () 经典实现(极简代码)

java 复制代码
/**
 * 线程等待打印:线程1先执行完毕,线程2再执行(Thread.join() 经典方案)
 * 核心:join() 让主线程阻塞等待线程1,再启动线程2
 */
public class WaitPrintByJoinDemo {
    public static void main(String[] args) {
        // 线程1:打印 1~3
        Thread thread1 = new Thread(() -> {
            for (int i = 1; i <= 3; i++) {
                System.out.println(Thread.currentThread().getName() + ":" + i);
                sleep(500); // 模拟耗时,让效果更直观
            }
            System.out.println(Thread.currentThread().getName() + " 执行完毕");
        }, "线程1");

        // 线程2:打印 A~C
        Thread thread2 = new Thread(() -> {
            for (char c = 'A'; c <= 'C'; c++) {
                System.out.println(Thread.currentThread().getName() + ":" + c);
                sleep(500); // 模拟耗时
            }
            System.out.println(Thread.currentThread().getName() + " 执行完毕");
        }, "线程2");

        // 步骤1:启动线程1
        thread1.start();

        // 步骤2:主线程等待线程1执行完毕(核心:join())
        try {
            thread1.join();
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
        }

        // 步骤3:线程1执行完毕后,再启动线程2
        thread2.start();
    }

    /**
     * 极简睡眠工具方法(避免重复 try-catch)
     */
    private static void sleep(long millis) {
        try {
            Thread.sleep(millis);
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
        }
    }
}

打印结果:

java 复制代码
线程1:1
线程1:2
线程1:3
线程1 执行完毕
线程2:A
线程2:B
线程2:C
线程2 执行完毕

方案 B:CompletableFuture 优雅实现(JDK8+)

java 复制代码
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.TimeUnit;

/**
 * 线程等待打印:线程1先执行完毕,线程2再执行(CompletableFuture 优雅方案)
 * 核心:thenRunAsync() 实现任务依赖,无需手动管理线程和join()
 */
public class WaitPrintByCompletableFutureDemo {
    public static void main(String[] args) {
        // 任务1(对应线程1):打印 1~3
        CompletableFuture<Void> task1 = CompletableFuture.runAsync(() -> {
            for (int i = 1; i <= 3; i++) {
                System.out.println("线程1:" + i);
                sleep(500);
            }
            System.out.println("线程1 执行完毕");
        });

        // 任务2(对应线程2):依赖任务1执行完毕后执行,打印 A~C
        CompletableFuture<Void> task2 = task1.thenRunAsync(() -> {
            for (char c = 'A'; c <= 'C'; c++) {
                System.out.println("线程2:" + c);
                sleep(500);
            }
            System.out.println("线程2 执行完毕");
        });

        // 主线程等待任务2执行完毕(可选,根据需求决定)
        task2.join();
    }

    /**
     * 极简睡眠工具方法
     */
    private static void sleep(long millis) {
        try {
            TimeUnit.MILLISECONDS.sleep(millis);
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
        }
    }
}

打印结果:

java 复制代码
线程1:1
线程1:2
线程1:3
线程1 执行完毕
线程2:A
线程2:B
线程2:C
线程2 执行完毕
相关推荐
Drifter_yh3 小时前
【黑马点评】Redisson 分布式锁核心原理剖析
java·数据库·redis·分布式·spring·缓存
莫寒清4 小时前
Spring MVC:@RequestParam 注解详解
java·spring·mvc
没有医保李先生5 小时前
字节对齐的总结
java·开发语言
甲枫叶6 小时前
【claude】Claude Code正式引入Git Worktree原生支持:Agent全面实现并行独立工作
java·人工智能·git·python·ai编程
六件套是我6 小时前
无法访问org.springframeword.beans.factory.annotation.Value
java·开发语言·spring boot
LYS_06187 小时前
C++学习(5)(函数 指针 引用)
java·c++·算法
forestsea7 小时前
Spring Cloud Alibaba 2025.1.0.0 正式发布:拥抱 Spring Boot 4.0 与 Java 21+ 的新时代
java·spring boot·后端
IT枫斗者7 小时前
IntelliJ IDEA 2025.3史诗级更新:统一发行版+Spring Boot 4支持,这更新太香了!
java·开发语言·前端·javascript·spring boot·后端·intellij-idea
forestsea7 小时前
Spring Boot 4.0 + JDK 25 + GraalVM:下一代云原生Java应用架构
java·spring boot·云原生