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 执行完毕
相关推荐
pqq的迷弟7 小时前
面试整理:HashMap\ConcurrentHashMap原来
java·面试·职场和发展
夕除7 小时前
javaweb--16
java·状态模式
用户游民7 小时前
Android 腾讯X5WebView如何禁止系统自带剪切板和自定义剪切板视图
android·java
花月C7 小时前
Agent应用开发零基础入门:核心概念、环境配置与首次LLM调用
java·python
曹牧7 小时前
Java Web:DispatcherServlet
java·开发语言·前端
直奔標竿7 小时前
Java开发者AI转型第二十三课!Spring AI个人知识库实战(二):异步ETL流水线搭建与避坑指南
java·人工智能·spring boot·后端·spring
Lyyaoo.7 小时前
TreadLocal和TreadLocalMap
android·java·redis
AC赳赳老秦7 小时前
网安工程师提效:用 OpenClaw 实现漏洞扫描报告生成、安全巡检自动化、日志合规审计
java·开发语言·前端·javascript·python·deepseek·openclaw
ffqws_8 小时前
MyBatis 动态 SQL 详解:从原理到实战
java·sql·mybatis
浮尘笔记8 小时前
在Snowy后台无需编码实现自动化生成CRUD操作流程
java·开发语言·经验分享·spring boot·后端·程序人生·mybatis