LeetCode 1114. 按序打印

目录

[LeetCode 1114. 按序打印:多线程顺序控制的核心逻辑(附完整代码)](#LeetCode 1114. 按序打印:多线程顺序控制的核心逻辑(附完整代码))

一、题目核心需求

[二、解题核心思路:锁 + 标志位 + 线程通信](#二、解题核心思路:锁 + 标志位 + 线程通信)

三、完整代码实现

四、关键知识点总结


LeetCode 1114. 按序打印:多线程顺序控制的核心逻辑(附完整代码)

最近练习了 LeetCode 的「按序打印」问题,正好借此梳理多线程顺序控制的核心思路 ------ 这是并发编程的基础考点,分享给刚学 JUC 的朋友~

一、题目核心需求

实现一个Foo类,让三个线程分别调用first()second()third()严格按first→second→third的顺序打印(即使线程启动顺序打乱)。

二、解题核心思路:锁 + 标志位 + 线程通信

要让多线程按固定顺序执行,核心是 **"控制前置方法的执行状态"**,这里用「synchronized锁 + 布尔标志位 + wait/notifyAll通信」实现:

  1. 标志位 :用firsttestsecondtest标记 "前序方法是否执行完成";
  2. 专属锁对象 :创建private final Object lock作为同步锁(避免外部干扰);
  3. 线程通信 :用lock.wait()让线程等待(释放锁),lock.notifyAll()唤醒后续线程。

三、完整代码实现

复制代码
class Foo {
    // 标志位:标记前序方法是否执行完成
    private boolean firsttest = false;
    private boolean secondtest = false;
    // 专属锁对象:保证锁的封闭性和唯一性
    private final Object lock = new Object();

    public Foo() {
    }

    public void first(Runnable printFirst) throws InterruptedException {
        synchronized (lock) {
            // 执行first逻辑
            printFirst.run();
            // 标记first已完成
            firsttest = true;
            // 唤醒等待lock的线程(second/third)
            lock.notifyAll();
        }
    }

    public void second(Runnable printSecond) throws InterruptedException {
        synchronized (lock) {
            // 循环等待:直到first执行完成(防止虚假唤醒)
            while (!firsttest) {
                lock.wait();
            }
            // 执行second逻辑
            printSecond.run();
            // 标记second已完成
            secondtest = true;
            // 唤醒等待lock的线程(third)
            lock.notifyAll();
        }
    }

    public void third(Runnable printThird) throws InterruptedException {
        synchronized (lock) {
            // 循环等待:直到second执行完成
            while (!secondtest) {
                lock.wait();
            }
            // 执行third逻辑
            printThird.run();
        }
    }
}

四、关键知识点总结

  1. 专属锁对象的设计private final Object lock而不是this

    • private:避免外部代码滥用锁;
    • final:防止锁对象被意外替换;
    • Object:轻量化,仅用其 "对象身份" 做锁。
  2. 为什么用while而不是if判断标志位? 防止虚假唤醒 (线程可能无原因被唤醒),while会二次检查标志位,确保只有前序方法执行完才继续。

  3. wait/notifyAll的作用

    • lock.wait():释放锁,让前序线程执行;
    • lock.notifyAll():唤醒所有等待该锁的线程,传递 "前序已完成" 的信号。
相关推荐
云烟成雨TD22 分钟前
Spring AI Alibaba 1.x 系列【69】Token 用量统计
java·人工智能·spring
Navigator_Z24 分钟前
LeetCode //C - 1089. Duplicate Zeros
c语言·算法·leetcode
JAVA96526 分钟前
JAVA面试-并发篇 03-使用synchronized doublecheck实现单例有什么坑
java·单例模式·面试
在繁华处28 分钟前
Java从零到熟练(四):面向对象基础
java·开发语言
Unbelievabletobe30 分钟前
解决了股票api接口盘后数据更新慢的问题
大数据·开发语言·python
不会C语言的男孩2 小时前
C++ Primer 第2章:变量和基本类型
开发语言·c++
小江的记录本2 小时前
【JVM虚拟机】堆内存分代模型:年轻代(Eden+Survivor)、老年代、元空间Metaspace(附《思维导图》+《面试高频考点清单》)
java·前端·jvm·后端·python·spring·面试
在繁华处2 小时前
Java从零到熟练(三):流程控制
java·开发语言·python
唐青枫2 小时前
Java Optional 实战指南:优雅处理空值与链式转换
java
一起学开源2 小时前
一文读懂 ReAct 范式:让 AI Agent 真正学会“思考+行动“
java·javascript·react.js·ecmascript·react·alibaba·智能体开发