多线程(69)如何使用synchronized实现信号量

在Java中,信号量(Semaphore)是一种常用的同步工具,它可以用来控制对共享资源的访问数量。信号量主要用于两个目的:一个是用于多个共享资源的互斥使用,另一个是用于并发线程数的控制。虽然Java的java.util.concurrent包提供了Semaphore类,但了解如何手动实现一个信号量可以帮助更深入地理解并发编程的原理。

下面,我们将使用Synchronized关键字来实现一个简单的信号量。我们的目标是实现一个计数信号量,其中信号量的计数指示可以同时访问某一资源的线程数。

实现基本框架

首先,我们定义一个Semaphore类,它需要维护一个计数器来跟踪可用的许可证数量。计数器的初始值在信号量创建时通过构造函数提供。

java 复制代码
public class SimpleSemaphore {
    private int signals = 0;
    private int bound = 0;

    public SimpleSemaphore(int upperBound) {
        this.bound = upperBound;
    }
}

实现acquire方法

acquire方法用于获取一个许可。如果当前没有可用的许可(即signals等于上界bound),那么该方法将阻塞,直到有许可可用。

java 复制代码
public synchronized void acquire() throws InterruptedException {
    while (signals == bound) {
        wait();
    }
    signals++;
    notify();
}

这里使用synchronized关键字来保证方法是线程安全的。如果signals达到bound,调用wait()使当前线程等待,直到signals减少后再继续执行。每次调用acquire方法时,signals都会增加,并且通过notify()唤醒可能在等待的线程。

实现release方法

release方法用于释放一个许可,增加可用许可的数量。如果有线程正在等待许可,那么其中一个线程将被唤醒。

java 复制代码
public synchronized void release() throws InterruptedException {
    while (signals == 0) wait();
    signals--;
    notify();
}

同样,使用synchronized确保线程安全,并在释放许可后通过notify()唤醒可能正在等待的线程。

示例使用

以下是如何使用SimpleSemaphore的一个简单示例。

java 复制代码
public class Main {
    public static void main(String[] args) {
        SimpleSemaphore semaphore = new SimpleSemaphore(3); // 允许3个线程同时访问

        Runnable longRunningTask = () -> {
            try {
                semaphore.acquire();
                System.out.println("Thread " + Thread.currentThread().getId() + " is running");
                // 模拟长时间的任务
                Thread.sleep(2000);
                semaphore.release();
                System.out.println("Thread " + Thread.currentThread().getId() + " is finished");
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        };

        for (int i = 0; i < 10; i++) {
            new Thread(longRunningTask).start();
        }
    }
}

在这个示例中,我们创建了一个具有3个许可的SimpleSemaphore,然后启动了10个线程来执行一项长时间运行的任务。由于信号量的限制,这10个线程将分批(每批3个)执行。

总结

通过使用synchronized关键字、wait()notify()方法,我们可以手动实现一个简单的信号量。这个实现提供了互斥访问和线程间的协调能力。虽然Java的Semaphore类提供了更高级的功能,但手动实现信号量是理解并发控制的一个很好的练习。记住,真实环境下应优先使用Java标准库中的并发工具,因为它们经过了更广泛的测试并且优化得更好。

相关推荐
咚为7 小时前
Rust Print 终极指南:从底层原理到全场景实战
开发语言·后端·rust
二哈喇子!8 小时前
基于Spring Boot框架的车库停车管理系统的设计与实现
java·spring boot·后端·计算机毕业设计
Loo国昌11 小时前
【LangChain1.0】第九阶段:文档处理工程 (LlamaIndex)
人工智能·后端·python·算法·langchain
毕设源码-朱学姐11 小时前
【开题答辩全过程】以 基于SpringBoot的律师事务所管理系统的设计与实现为例,包含答辩的问题和答案
java·spring boot·后端
愈努力俞幸运12 小时前
flask 入门 token, headers,cookie
后端·python·flask
毕设源码-朱学姐12 小时前
【开题答辩全过程】以 基于springboot的日用药品仓库管理系统的设计与实现为例,包含答辩的问题和答案
java·spring boot·后端
lkbhua莱克瓦2413 小时前
深入理解HTTP协议:从理论到SpringBoot实践
网络·笔记·后端·网络协议·http·javaweb
古城小栈14 小时前
Rust复合类型 四大军阀:数、元、切、串
开发语言·后端·rust
+VX:Fegn089515 小时前
计算机毕业设计|基于springboot + vue酒店预订系统(源码+数据库+文档)
数据库·vue.js·spring boot·后端·课程设计
JavaGuide15 小时前
IntelliJ IDEA 2026.1 EAP 发布!拥抱 Java 26,Spring Boot 4 深度支持!
java·后端·mysql·springboot·idea·大厂面试·javaguide