0

0

Java并发编程:使用Semaphore实现线程交替执行的精确控制

聖光之護

聖光之護

发布时间:2025-11-09 16:09:01

|

351人浏览过

|

来源于php中文网

原创

java并发编程:使用semaphore实现线程交替执行的精确控制

本文深入探讨了在Java中使用Semaphore实现两个线程交替、顺序执行特定任务的机制。通过分析一个常见的编程错误——即线程未能共享同一个Semaphore实例,导致同步失效——我们展示了如何正确地初始化和共享Semaphore,以确保线程之间能够有效协调,从而实现“121212...”这样的精确输出序列。

在多线程编程中,确保线程以特定顺序执行任务是常见的需求。Java提供了多种同步机制来协调线程间的操作,其中Semaphore(信号量)是一种强大的工具,它允许我们控制对共享资源的并发访问数量。本文将详细介绍如何利用Semaphore实现两个线程的严格交替执行,并纠正一个在实践中容易犯的错误。

线程交替执行的需求与Semaphore原理

设想一个场景:我们需要两个线程,一个打印数字“1”(我们称之为P1),另一个打印数字“2”(P2)。要求它们的输出严格按照“121212...”的顺序交替进行。这意味着P1必须先打印“1”,然后P2才能打印“2”,P2完成后P1才能再次打印“1”,如此循环。

Semaphore通过维护一个许可计数器来工作。acquire()方法会尝试获取一个许可,如果计数器为零,线程将被阻塞直到有许可可用。release()方法会释放一个许可,增加计数器。通过巧妙地初始化和操作Semaphore的许可,我们可以实现线程间的顺序控制。

立即学习Java免费学习笔记(深入)”;

为了实现“1212”的交替打印,我们可以使用两个Semaphore:

  • sem1:初始许可为1,用于控制P1线程的执行。
  • sem2:初始许可为0,用于控制P2线程的执行。

具体流程如下:

Sora
Sora

Sora是OpenAI发布的一种文生视频AI大模型,可以根据文本指令创建现实和富有想象力的场景。

下载
  1. P1线程尝试获取sem1的许可。由于sem1初始许可为1,P1可以立即获取并执行。
  2. P1打印“1”后,释放sem2的许可。
  3. P2线程尝试获取sem2的许可。由于P1刚刚释放了sem2的许可,P2可以立即获取并执行。
  4. P2打印“2”后,释放sem1的许可。
  5. 循环往复,P1再次获取sem1的许可,形成交替执行。

常见错误:Semaphore实例未共享

在实现上述逻辑时,一个非常常见的错误是,不同的线程实例操作的是各自独立的Semaphore实例,导致它们之间无法进行有效的同步。考虑以下错误示例代码:

import java.util.concurrent.Semaphore;

public class SemTestIncorrect {
    // 每个SemTestIncorrect实例都会有自己的sem1和sem2
    Semaphore sem1 = new Semaphore(1);
    Semaphore sem2 = new Semaphore(0);

    public static void main(String args[]) {
        // 错误:创建了两个SemTestIncorrect实例
        // 每个实例都有自己独立的sem1和sem2
        final SemTestIncorrect semTest1 = new SemTestIncorrect();
        final SemTestIncorrect semTest2 = new SemTestIncorrect();

        new Thread() {
            @Override
            public void run() {
                try {
                    semTest1.numb1(); // 线程1操作semTest1的semaphores
                } catch (Exception e) {
                    throw new RuntimeException(e);
                }
            }
        }.start();

        new Thread() {
            @Override
            public void run() {
                try {
                    semTest2.numb2(); // 线程2操作semTest2的semaphores
                } catch (Exception e) {
                    throw new RuntimeException(e);
                }
            }
        }.start();
    }

    private void numb1() {
        while (true) {
            try {
                sem1.acquire(); // 获取当前实例的sem1
                System.out.print(" 1");
                sem2.release(); // 释放当前实例的sem2
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }

    private void numb2() {
        while (true) {
            try {
                sem2.acquire(); // 获取当前实例的sem2
                System.out.print(" 2");
                sem1.release(); // 释放当前实例的sem1
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}

上述代码的问题在于 main 方法中创建了两个 SemTestIncorrect 对象:semTest1 和 semTest2。每个对象都拥有自己独立的 sem1 和 sem2 信号量实例。第一个线程调用 semTest1.numb1(),操作的是 semTest1 内部的 sem1 和 sem2。而第二个线程调用 semTest2.numb2(),操作的是 semTest2 内部的 sem1 和 sem2。由于这两个线程操作的是不同的信号量对,它们之间无法进行任何通信和同步,导致程序在打印一个“1”之后就停止(因为 semTest1 释放了它自己的 sem2,但 semTest2 的 sem2 仍然为0,无法被获取)。

正确的解决方案:共享Semaphore实例

要解决这个问题,关键在于确保所有参与同步的线程都操作同一个Semaphore实例。这可以通过多种方式实现,例如:

  1. 将Semaphore声明为static成员,使其成为类级别的共享资源。
  2. 在main方法中创建Semaphore实例,并通过构造函数或方法参数传递给线程或其执行的Runnable/Callable。
  3. 只创建一个包含Semaphore的类实例,并让所有线程都调用该实例的方法。

以下是采用第三种方式的修正代码,这也是最直接且符合面向对象原则的修正:

import java.util.concurrent.Semaphore;

public class SemTestCorrect {
    // Semaphores现在属于这个类的实例
    private final Semaphore sem1 = new Semaphore(1); // P1先执行
    private final Semaphore sem2 = new Semaphore(0); // P2后执行

    public static void main(String args[]) {
        // 正确:只创建一个SemTestCorrect实例
        // 两个线程将操作这同一个实例中的sem1和sem2
        final SemTestCorrect sharedSemTest = new SemTestCorrect();

        new Thread(() -> { // 使用Lambda表达式简化线程创建
            try {
                sharedSemTest.numb1(); // 线程1操作共享实例的semaphores
            } catch (Exception e) {
                throw new RuntimeException(e);
            }
        }).start();

        new Thread(() -> { // 使用Lambda表达式简化线程创建
            try {
                sharedSemTest.numb2(); // 线程2操作共享实例的semaphores
            } catch (Exception e) {
                throw new RuntimeException(e);
            }
        }).start();
    }

    public void numb1() {
        while (true) {
            try {
                sem1.acquire(); // P1获取sem1许可
                System.out.print(" 1");
                sem2.release(); // P1释放sem2许可,允许P2执行
                Thread.sleep(500); // 模拟工作耗时
            } catch (InterruptedException e) {
                Thread.currentThread().interrupt(); // 重新设置中断状态
                System.out.println("P1 interrupted.");
                break; // 退出循环
            }
        }
    }

    public void numb2() {
        while (true) {
            try {
                sem2.acquire(); // P2获取sem2许可
                System.out.print(" 2");
                sem1.release(); // P2释放sem1许可,允许P1执行
                Thread.sleep(500); // 模拟工作耗时
            } catch (InterruptedException e) {
                Thread.currentThread().interrupt(); // 重新设置中断状态
                System.out.println("P2 interrupted.");
                break; // 退出循环
            }
        }
    }
}

在这段修正后的代码中,main方法只创建了一个 SemTestCorrect 实例 sharedSemTest。两个线程都通过这个同一个 sharedSemTest 实例来调用 numb1() 和 numb2() 方法。这样,它们就能够访问并操作同一个 sem1 和 sem2 信号量对象,从而实现正确的交替执行。输出将是连续的“1 2 1 2 1 2...”。

注意事项与总结

  1. 共享资源原则: 任何用于线程间同步的机制(如Semaphore, Lock, Condition, volatile变量等)都必须是所有相关线程能够访问到的同一个实例。这是多线程编程中最基本也是最重要的原则之一。
  2. InterruptedException处理: 当线程在等待获取许可时被中断,acquire()方法会抛出InterruptedException。正确的处理方式是捕获异常,并通常重新设置线程的中断状态 (Thread.currentThread().interrupt();),然后决定是继续执行还是退出循环。
  3. Semaphore与Mutex: 当Semaphore的许可数量设置为1时,它实际上充当了一个二元信号量,功能类似于互斥锁(Mutex)。Java中的ReentrantLock是更常用的互斥锁实现,它提供了更丰富的锁定功能,例如条件变量(Condition)。对于这种严格的交替执行模式,使用两个Semaphore进行信号传递是一种简洁有效的方法。如果使用ReentrantLock,通常需要配合Condition对象来完成类似的等待/通知机制。
  4. 死锁风险: 在复杂的同步场景中,不当的Semaphore使用可能导致死锁。例如,如果线程A持有Semaphore X并尝试获取Semaphore Y,而线程B持有Semaphore Y并尝试获取Semaphore X,就可能发生死锁。本例中的设计避免了这种风险,因为每个线程只尝试获取一个特定的Semaphore,并在完成后释放另一个。

通过本文的讲解和修正后的代码示例,我们理解了如何正确利用Java的Semaphore实现线程间的精确交替执行,并强调了共享同步资源的重要性。掌握这些基本概念对于编写健壮、高效的并发程序至关重要。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

836

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

741

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

736

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

397

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

399

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

430

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16926

2023.08.03

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

72

2026.01.16

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.6万人学习

C# 教程
C# 教程

共94课时 | 7万人学习

Java 教程
Java 教程

共578课时 | 47.5万人学习

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

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