0

0

Semaphore 实现线程交替执行的正确用法详解

聖光之護

聖光之護

发布时间:2026-03-06 18:17:01

|

417人浏览过

|

来源于php中文网

原创

Semaphore 实现线程交替执行的正确用法详解

本文详解如何使用 Java Semaphore 实现两个线程严格交替执行(如输出 "FooBarFooBar..."),指出原代码因串行调用导致失效的根本原因,并提供可运行、线程安全的完整解决方案。

本文详解如何使用 java `semaphore` 实现两个线程严格交替执行(如输出 "foobarfoobar..."),指出原代码因串行调用导致失效的根本原因,并提供可运行、线程安全的完整解决方案。

在多线程协作场景中,常需控制多个线程按特定顺序交替执行——例如实现 FooBar 交替打印。Semaphore(信号量)是实现此类同步逻辑的理想工具:它通过许可(permit)的获取与释放,精确协调线程执行节奏。但若使用不当(如在主线程中串行调用两个方法),将完全丧失并发性,导致预期失效。

? 核心问题:原代码为何不工作?

原始代码中,main 方法内依次调用:

pt.foo(printFo); // 阻塞式执行完全部5次Foo
pt.bar(printBa); // 再执行全部5次Bar

这本质上是单线程串行执行,两个 Runnable 并未并发运行,Semaphore 的 acquire()/release() 完全失去意义,自然无法实现交替。

真正有效的方案必须满足:

Google Maps Agent
Google Maps Agent

Google地图应用构建智能体工具

下载
  • ✅ foo() 和 bar() 在独立线程中并发启动;
  • ✅ Semaphore 初始状态合理:foo 先行 → f = new Semaphore(1)(允许首次执行),t = new Semaphore(0)(阻塞 bar 直到 foo 首次释放);
  • ✅ 每次执行后立即释放对方许可,形成“你放我、我放你”的闭环。

✅ 正确实现:双线程 + 协作信号量

以下是精简、健壮、符合最佳实践的完整代码:

import java.util.concurrent.Semaphore;

public class PrintThread {
    private final int n = 5;
    private final Semaphore fooPermit = new Semaphore(1);   // 初始允许 foo 执行
    private final Semaphore barPermit = new Semaphore(0);   // 初始阻塞 bar

    public void foo(Runnable printFoo) {
        for (int i = 0; i < n; i++) {
            try {
                fooPermit.acquire(); // 等待轮到自己
                printFoo.run();     // 打印 "Foo"
                barPermit.release(); // 通知 bar 可执行
            } catch (InterruptedException e) {
                Thread.currentThread().interrupt();
                throw new RuntimeException("Foo thread interrupted", e);
            }
        }
    }

    public void bar(Runnable printBar) throws InterruptedException {
        for (int i = 0; i < n; i++) {
            barPermit.acquire(); // 等待 foo 释放许可
            printBar.run();      // 打印 "Bar"
            fooPermit.release(); // 通知 foo 可执行
        }
    }

    public static void main(String[] args) throws Exception {
        PrintThread pt = new PrintThread();
        Runnable printFoo = () -> System.out.print("Foo");
        Runnable printBar = () -> System.out.print("Bar");

        Thread t1 = new Thread(() -> pt.foo(printFoo));
        Thread t2 = new Thread(() -> {
            try {
                pt.bar(printBar);
            } catch (InterruptedException e) {
                Thread.currentThread().interrupt();
            }
        });

        t1.start();
        t2.start();

        // 等待两线程完成(更健壮做法:使用 CountDownLatch)
        t1.join();
        t2.join();

        System.out.println(); // 换行,便于观察输出
    }
}

预期输出:

FooBarFooBarFooBarFooBarFooBar

⚠️ 关键注意事项

  • 勿忽略中断处理:acquire() 可能被中断,务必捕获 InterruptedException 并恢复中断状态(Thread.currentThread().interrupt()),避免掩盖线程取消意图。
  • 避免 Thread.sleep() 硬等待:原答案中 Thread.sleep(1000) 不可靠(可能过早结束或过度延迟)。应使用 join() 或 CountDownLatch 精确同步主线程与工作线程。
  • 语义清晰命名:将 f/t 改为 fooPermit/barPermit,大幅提升可读性与可维护性。
  • final 修饰符增强安全性:n、Semaphore 实例声明为 final,明确其不可变性,符合并发编程习惯。

✅ 总结

Semaphore 实现线程交替的本质是状态驱动的许可传递:一个线程执行后释放另一个线程所需的许可,形成确定性的执行链。成功的关键在于:

  1. 并发执行(启用独立线程),
  2. 初始许可配置正确(谁先谁后需匹配 acquire/release 顺序),
  3. 每次操作原子闭环(acquire → work → release 缺一不可)。

掌握此模式,即可灵活扩展至 FooBarBaz 三线程循环、生产者-消费者配对等更复杂协同场景。

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

743

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

375

2025.12.24

java多线程相关教程合集
java多线程相关教程合集

本专题整合了java多线程相关教程,阅读专题下面的文章了解更多详细内容。

27

2026.01.21

C++多线程相关合集
C++多线程相关合集

本专题整合了C++多线程相关教程,阅读专题下面的的文章了解更多详细内容。

28

2026.01.21

C# 多线程与异步编程
C# 多线程与异步编程

本专题深入讲解 C# 中多线程与异步编程的核心概念与实战技巧,包括线程池管理、Task 类的使用、async/await 异步编程模式、并发控制与线程同步、死锁与竞态条件的解决方案。通过实际项目,帮助开发者掌握 如何在 C# 中构建高并发、低延迟的异步系统,提升应用性能和响应速度。

103

2026.02.06

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

375

2025.12.24

java多线程相关教程合集
java多线程相关教程合集

本专题整合了java多线程相关教程,阅读专题下面的文章了解更多详细内容。

27

2026.01.21

C++多线程相关合集
C++多线程相关合集

本专题整合了C++多线程相关教程,阅读专题下面的的文章了解更多详细内容。

28

2026.01.21

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

1

2026.03.06

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号