信号量(Semaphore)与线程计数器(CountDownLatch)(详解)

🍉信号量(Semaphore)

Semaphore属于共享锁,即多个线程可以同时获取,用来表示可用资源的个数,本质上是一个计数器

🥩理解信号量:

🍂我们将信号量理解为一个停车场的空车位,例如当前有100个空车位,表示100个可用资源 🍂当有车开进停车场,就相当于申请一个可用资源,空车位就-1(这个称为信号量的P操作) 🍂当有车开出停车场,就相当于释放一个可用资源,空车位就+1(这个称为信号量的V操作) 🍂如果可用资源为0,会尝试申请资源,就会阻塞等待,直到有其他线程释放资源
🥩注意: Semaphore的PV操作的加减计数器操作都是原子性的,可以直接在多线程环境下使用

🍡Semaphore的构造方法:

🍭Semaphore的常用方法:

🍢使用场景:

☘️等待一组线程执行完,再执行某个任务

☘️同一个时间最多执行n个线程(有限资源的使用)

🍴示例:

创建Semaphore实例,初始化为4,表示4个可用资源

acquire方法表示申请资源(P操作),release方法表示释放资源(V操作)

创建20个线程,每个线程都尝试申请资源,sleep等待1秒后,释放资源,观察程序执行结果

👁‍🗨️代码展示:

java 复制代码
import java.util.concurrent.Semaphore;
 
public class SemaphoreTest {
    public static void main(String[] args) {
        Semaphore sem = new Semaphore(4);
        Runnable runnable = new Runnable() {
            @Override
            public void run() {
                try {
                    System.out.println("申请资源");
                    sem.acquire();
                    System.out.println("获取到资源");
                    Thread.sleep(1000);
                    sem.release();
                    System.out.println("释放资源");
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        };
        for(int i = 0;i < 20;i++){
            Thread t = new Thread(runnable);
            t.start();
        }
    }
}

👁‍🗨️打印结果说明:

🥭线程计数器(CountDownLatch)

CountDownLatch也属于共享锁,其内部有一个int类型的属性表示可以同时并发并行的线程的数量

同时等待N个任务执行结束

举例说明:

比如跑步比赛,必须等所有运动员通过终点才能公布成绩

🍭CountDownLatch的构造方法:

🍡CountDownLatch的常用方法:

🥩使用场景: 等待多个线程全部执行完,再执行某个任务
🥩注意: CountDownLatch只能减不能加

🍴示例:

构造CountDownLatch实例,初始化为10,表示有10个任务需要完成

每个任务执行完成后,调用countDown(),CountDownLatch内部计数器自减

主线程调用await(),等待所有线程执行完毕,也就是计数器值为0,再继续执行主线程后续任务

👁‍🗨️代码展示:

java 复制代码
import java.util.concurrent.CountDownLatch;
 
public class CountDownLatchTest {
    public static void main(String[] args) throws InterruptedException {
        CountDownLatch latch = new CountDownLatch(10);
        Runnable runnable = new Runnable() {
            @Override
            public void run() {
                System.out.println(Thread.currentThread().getName());
                latch.countDown();
            }
        };
        for(int i = 0;i < 10;i++){
            Thread t = new Thread(runnable);
            t.start();
        }
        latch.await();
        System.out.println("10个线程比赛结束");
    }
}

👁‍🗨️打印结果说明:

相关推荐
追逐时光者7 小时前
一个致力于为 C# 程序员提供更佳的编码体验和效率的 Visual Studio 扩展插件
后端·c#·visual studio
韩师学子--小倪8 小时前
fastjson与gson的toString差异
java·json
Drawing stars8 小时前
JAVA后端 前端 大模型应用 学习路线
java·前端·学习
nbsaas-boot8 小时前
SQL Server 存储过程开发规范(公司内部模板)
java·服务器·数据库
行百里er9 小时前
用 ThreadLocal + Deque 打造一个“线程专属的调用栈” —— Spring Insight 的上下文管理术
java·后端·架构
玄〤9 小时前
黑马点评中 VoucherOrderServiceImpl 实现类中的一人一单实现解析(单机部署)
java·数据库·redis·笔记·后端·mybatis·springboot
J_liaty9 小时前
Spring Boot拦截器与过滤器深度解析
java·spring boot·后端·interceptor·filter
短剑重铸之日9 小时前
《7天学会Redis》Day2 - 深入Redis数据结构与底层实现
数据结构·数据库·redis·后端
码事漫谈10 小时前
从C++到C#的转型完全指南
后端
好好沉淀10 小时前
1.13草花互动面试
面试·职场和发展