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 执行完毕
相关推荐
80530单词突击赢2 小时前
云原生时代:.NET与Java的K8s进化论
java
hhy_smile2 小时前
Special method in class
java·开发语言
我命由我123452 小时前
Android 开发 Room 数据库升级问题:A migration from 6 to 7 was required but not found.
android·java·java-ee·android studio·android jetpack·android-studio·android runtime
黄筱筱筱筱筱筱筱2 小时前
7.适合新手小白学习Python的异常处理(Exception)
java·前端·数据库·python
Stecurry_302 小时前
Springboot整合SpringMVC --从0到1
java·spring boot·后端
Serene_Dream2 小时前
NIO 的底层机理
java·jvm·nio·mmap
skywalker_112 小时前
多线程&JUC
java·开发语言·jvm·线程池
黎雁·泠崖2 小时前
Java基础核心能力总结:从语法到API的完整知识体系
java·开发语言
_周游2 小时前
Java8 API 文档搜索引擎_2.索引模块(实现细节)
java·搜索引擎·intellij-idea