【多线程进阶】JUC下的常用类

news2024/11/17 19:40:11

🎉🎉🎉点进来你就是我的人了
博主主页:🙈🙈🙈戳一戳,欢迎大佬指点!

欢迎志同道合的朋友一起加油喔🤺🤺🤺


目录

ReentrantLock

Semaphore(信号量)

计数器——CountDownLatch



"JUC"是"Java Util Concurrent"的缩写,代表Java提供的一套并发工具类。这些工具类大大简化了编程并发和多线程应用的复杂性,提供了更高级、更强大、更安全的并发操作功能。以下是一些常见的JUC类:

  1. ExecutorServiceThreadPoolExecutorExecutors:这些类用于线程池的创建和管理。它们可以控制线程的数量,自动管理线程的生命周期,并提供将任务提交到线程池执行的方法。

  2. FutureCallableFuture表示异步计算的结果,Callable则是具有返回值的Runnable,用于在ExecutorService中执行任务。

  3. CountDownLatch:一个同步辅助类,在完成一组正在其它线程中执行的操作之前,它允许一个或多个线程一直等待。

  4. CyclicBarrier:它允许一组线程互相等待,直到所有线程都达到一个公共的屏障点。

  5. Semaphore:一个计数信号量,常用于限制可以访问某些资源(物理或逻辑的)的线程数目。

  6. ConcurrentHashMap:一个线程安全的HashMap。

  7. AtomicIntegerAtomicLongAtomicReference:原子变量类,用于实现无锁的线程安全操作。

  8. LockReentrantLockCondition:显示锁和条件对象,用于实现比synchronized更复杂的线程同步和通信。

此处主要介绍一下之前没有提到的类 (其它类的使用可以在我多线程的专栏查看)

ReentrantLock

ReentrantLock是标准库为我们提供的另一种锁,顾名思义。该锁也是可重入锁。
我们的Synchronized也是我们标准库提供的锁,那么它们有什么区别呢?
Synchronized: 是直接基于代码块进行加锁解锁的。
ReentrantLock:是使用了lock()方法和unlock()方法进行加锁解锁的。

方法作用
lock()加锁,获取不到就死等
trylock(time 超时时间)加锁,如果一定时间内没有获取到就放弃
unlock()解锁

我们的ReentrantLock在使用起来可能有一些需要注意的事项:

a. lock写在try之前;
b.一定要在finaly里面进行unlock();
  1. lock() 方法应在 try 代码块之前调用。这是因为,如果 lock() 方法在 try 代码块内并且发生异常,那么锁可能无法正确获取,而在 finally 代码块中,unlock() 方法仍会被执行。这可能会尝试释放并未真正被线程持有的锁,从而引发 IllegalMonitorStateException 异常。

  2. unlock() 方法应在 finally 代码块中调用,以确保锁一定会被释放,无论 try 代码块中的代码是否抛出异常。这是因为,如果 try 代码块中的代码抛出异常,并且 unlock() 不在 finally 中被执行,那么可能导致锁没有被正确释放,从而阻止其他线程获取该锁,进一步可能导致死锁现象。

上述是我们ReentrantLock的劣势,当然我们ReentrantLock也是有一些优势的。

1.我们的ReentrantKLock提供了公平锁版本的实现,我们的Synchronized只实现了非公平锁。

2. 我们Synchronized尝试加锁,如果锁已经被占有,就进行阻塞等待(死等),ReentrantLock提供了更灵活的获取锁的方式: trylock()

方法作用
trylock()无参数,能加锁就加,加不上就放弃
trylock(time)有参数,超过指定时间,加不上锁就放弃

3.ReentrantLock提供了一个更方便的等待通知机制,Synchronized搭配的是wait,notify,当我们notify的时候是随即唤醒一个wait状态的线程。ReentrantLock搭配一个Condition类,进行唤醒的时候可以唤醒指定线程。


Semaphore(信号量)

Semaphore(信号量):信号量本质就是一个计数操作,描述"可用资源的个数",这里设计两个操作。
P操作:申请一个可用资源,计数器-1
V操作:释放一个可用资源,计数器+1

这里我们的信号量可以搭配着停车场来理解:
我们的停车场的大牌子上就写着,当前剩余车位xx个,当有车进去,就相当于进行了一次P操作,计数器-1,当有车从停车场出来,就相当于进行了一次V操作,计数器+1,当我们的停车场的剩余车位为0时,也就是计数器为0时,如果有车想停车,要么在这里等,要么就去其他停车场。
我们的Semaphore同样提供了两个方法:

方法作用
acquire申请资源
release释放资源

我们在这里设计一个场景,我们图书馆一共有5本《算法导论》,但有10个同学去借

public static void main(String[] args) {
        Semaphore semaphore = new Semaphore(5);
        Runnable runnable = new Runnable() {
            @Override
            public void run() {
                try {
                    System.out.println("我来借书了");
                    semaphore.acquire();
                    System.out.println("我拿到算法导论了!");
                    Thread.sleep(500); //借阅时间
                    System.out.println("我还书了");
                    semaphore.release();
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
            }
        };
        for (int i = 0; i < 10; i++) {
            Thread t = new Thread(runnable);
            t.start();
        }
    }

我们的锁(可以认为是一个计数器为1的信号量),锁是信号量的特殊情况,信号量是锁的一般表达。
实际开发中,锁的出场频率是最高的,但也是有一些特殊的场景会用到信号量。


计数器——CountDownLatch

CountDownLatch是一个小组件,用于处理特殊场景下的。

就类似一场跑步比赛,开始时间是明确的,但结束时间是不确定的,只有所有选手冲线了才算结束。像这种场景,就引入了CountDownLatch.

方法作用
CountDownLatch(人数)在构造的时候,传入一个计数(参赛选手的个数)
await()等待所有线程执行结束
countDown()表示选手冲线

在开始时将CountDownLatch的计数器设定为参赛选手的数量,每个选手在跑完比赛后就调用countDown()方法将计数器减一。然后在比赛主管线程中,可以调用await()方法来等待所有选手完成比赛,也就是等待计数器变为0,此时比赛才算结束。 

调用await方法的线程,需要等待其他线程将计数器减为0,才能继续恢复执行。

import java.util.Random;
import java.util.concurrent.CountDownLatch;
 
/**
 * 使用CountDownLatch.await方法的线程会阻塞,直到所有等待线程全部执行结束为止
 * 大号进阶版本的join方法
 */
public class CountDownLatchTest {
    public static void main(String[] args) throws InterruptedException {
        // 等待线程需要等待的线程数,必须等这10个子线程全部执行完毕再恢复执行
        CountDownLatch latch = new CountDownLatch(10);
        Runnable runnable = new Runnable() {
            @Override
            public void run() {
                try {
                    Thread.sleep(new Random().nextInt(1000));
                    System.out.println(Thread.currentThread().getName() + "到达终点");
                    // 计数器 - 1
                    latch.countDown();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        };
        for (int i = 0; i < 10; i++) {
            Thread t = new Thread(runnable, "运动员" + i);
            t.start();
        }
        // main线程就是裁判线程,需要等待所有运动员到终点再恢复执行
        // 直到所有线程调用countdown方法将计数器减为0继续执行
        latch.await();
        System.out.println("比赛结束");
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/642028.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

阿里发布Java岗春招面试总结(复盘版)GitHub一经上传火爆全网!

Java面试都只是背答案吗? Java 八股文面试当然要背&#xff0c;不背就通过不了面试。 因为经济环境的问题&#xff0c;互联网行业竞争越来越严峻&#xff0c;面试也是越来越难。需要我们掌握的技术也是越来越多、要求是越来越高。作为一个资深的互联网码农&#xff0c;今天给…

公司招人面了一个00后测试,可以说是内卷届的天花板.....

公司前段缺人&#xff0c;也面了不少测试&#xff0c;结果竟然没有一个合适的。一开始瞄准的就是中级的水准&#xff0c;也没指望来大牛&#xff0c;提供的薪资也不低&#xff0c;面试的人很多&#xff0c;但平均水平很让人失望。令我印象最深的是一个00后测试员&#xff0c;他…

cpu、gpu的区别

1、CPU和GPU的内部构成 如上图所示&#xff0c;CPU和GPU的内部构成&#xff0c;可以看出是由不同的架构所组成 CPU&#xff1a;叫做中央处理器&#xff08;central processing unit&#xff09;作为计算机系统的运算和控制核心&#xff0c;是信息处理、程序运行的最终执行单元…

Splashtop 荣获两项“TrustRadius 2023年最受欢迎奖”

2023年5月10日 加利福尼亚州库比蒂诺 Splashtop 在简单易用的随处办公远程解决方案领域处于领先地位&#xff0c;该公司自豪地宣布&#xff0c;在远程桌面和远程支持这两个类别中荣获“TrustRadius 2023年最受欢迎奖”。这项知名奖项完全基于用户评论和客户情绪&#xff0c;强…

嵌入式软件实现定时器的两种方式

简介 在一般的嵌入式产品设计中&#xff0c;介于成本、功耗等&#xff0c;所选型的MCU基本都是资源受限的&#xff0c;而里面的定时器的数量更是有限。在我们软件设计中往往有多种定时需求&#xff0c;例如脉冲输出、按键检测、LCD切屏延时等等 &#xff0c;我们不可能让每一个…

JavaWeb__VUE

文章目录 概述快速入门Vue指令v-bindv-modelv-on条件判断v-for 概述 Vue是一套前端框架&#xff0c;免除原生JavaScript中的DOM操作&#xff0c;简化书写 基于MVVM(Model-View-ViewModel)思想&#xff0c;实现数据的双向绑定&#xff0c;将编程的关注点放在数据上。 Model是数…

ArduPilot之H743快速加减油门抬头现象分析

ArduPilot之H743快速加减油门抬头现象分析 1. 源由2. 现象分析3. 改善方法4. 验证5. 总结6. 参考资料 1. 源由 最近一直折腾再ArduCopter快速加减油四轴模型抬头的问题。 具体情况详见下面视频&#xff1a; 猛加油&#xff0c;机头后仰&#xff0c;然后点头&#xff1b; 快速…

一文搞清楚 Docker 镜像、容器、仓库

博主介绍&#xff1a; ✌博主从事应用安全和大数据领域&#xff0c;有8年研发经验&#xff0c;5年面试官经验&#xff0c;Java技术专家✌ Java知识图谱点击链接&#xff1a;体系化学习Java&#xff08;Java面试专题&#xff09; &#x1f495;&#x1f495; 感兴趣的同学可以收…

【服务器数据恢复】EXT3文件系统下Oracle数据库数据恢复案例

服务器数据恢复环境&#xff1a; 华为OceanStor某型号存储&#xff0c;十几块FC硬盘组建一组RAID5磁盘阵列&#xff0c;配备了一块热备盘&#xff1b;上层使用EXT3文件系统&#xff0c;配置了oracle数据库。 服务器故障&#xff1a; 该存储RAID5中的一块硬盘未知原因离线&…

从软件测试到自动化测试,待遇翻倍,我整理的超全学习指南!

因为我最近在分享自动化测试技术&#xff0c;经常被问到&#xff1a; 功能测试想转自动化&#xff0c;请问应该怎么入手&#xff1f;有没有好的资源推荐&#xff1f; 那么&#xff0c;接下来我就结合自己的经历聊一聊我是如何在工作中做自动化测试的。&#xff08;学习路线和…

教你该如何写单元测试

目录 前言&#xff1a; 到底什么是单元测试 为什么单测这么难写 写个单测例子 道阻且长 前言&#xff1a; 编写单元测试是软件开发中非常重要的一环&#xff0c;它可以确保代码的质量&#xff0c;减少Bug的产生&#xff0c;提高代码的可维护性&#xff0c;同时也能够大大…

ts自定义迭代器

key 为 [Symbol.iterator]

软考A计划-系统架构师-案例分析知识点整理

点击跳转专栏>Unity3D特效百例点击跳转专栏>案例项目实战源码点击跳转专栏>游戏脚本-辅助自动化点击跳转专栏>Android控件全解手册点击跳转专栏>Scratch编程案例点击跳转>软考全系列 &#x1f449;关于作者 专注于Android/Unity和各种游戏开发技巧&#xff…

【数据结构】24王道考研笔记——栈、队列和数组

三、栈、队列和数组 目录 三、栈、队列和数组栈基本概念顺序栈链式栈 队列基本概念顺序存储链式存储双端队列 应用括号匹配前中后缀表达式栈在递归中的运用队列的运用 数组数组的存储对称矩阵三角矩阵三对角矩阵稀疏矩阵 栈 基本概念 栈是只允许在一端进行插入或删除操作的线…

朋友拿下字节27K的offer,实名羡慕了....

最近有朋友去字节面试&#xff0c;面试前后进行了20天左右&#xff0c;包含4轮电话面试、1轮笔试、1轮主管视频面试、1轮hr视频面试。 据他所说&#xff0c;80%的人都会栽在第一轮面试&#xff0c;要不是他面试前做足准备&#xff0c;估计都坚持不完后面几轮面试。 其实&…

Redux异步解决方案 1. Redux-Thunk中间件

简单介绍一下thunk&#xff0c;这是一个中间件&#xff0c;是解决redux异步问题产生的。我们都知道&#xff0c;在使用redux的时候&#xff0c;通过dispatch一个action 发生到reducer 然后传递给store修改状态 一系列都是同步的&#xff0c;那如果说我dispatch一个action 这个a…

blockchain layer区块链分层

目录 1.layer0 2.layer1 3.layer2 ​4.layer3 1.layer0 第0层的定义目前行业还没有完全一致的理解。多数人认为第0层是 加密数据连接层及其硬件&#xff0c;对应上图下半部分。 也有一些人把跨链或可以创建链的基础设施为作为第0层&#xff0c;他们的代表有: LayerZero、S…

一文讲清后摩尔时代国产高性能并行应用软件生态建设

摘自《后摩尔时代国产高性能并行应用软件生态建设综述》 作者&#xff1a; 龚春叶1&#xff0c;刘杰1&#xff0c;包为民2&#xff0c;潘冬梅1&#xff0c;甘新标1&#xff0c;李胜国1 陈旭光1&#xff0c;肖调杰1&#xff0c;杨博1&#xff0c;王睿伯1 (1.国防科技大学 并…

基于stm32作品设计:多功能氛围灯、手机APP无线控制ws2812,MCU无线升级程序

文章目录 一、作品背景二、功能设计与实现过程三、实现基础功能&#xff08;一&#xff09;、首先是要选材&#xff08;二&#xff09;、原理图设计&#xff08;二&#xff09;、第一版本PCB设计&#xff08;三&#xff09;、焊接PCB板&#xff08;四&#xff09;编写单片机程序…

软件测试基础知识 —— 白盒测试

白盒测试 白盒测试&#xff08;White Box Testing&#xff09;又称结构测试、透明盒测试、逻辑驱动测试或基于代码的测试。白盒测试只测试软件产品的内部结构和处理过程&#xff0c;而不测试软件产品的功能&#xff0c;用于纠正软件系统在描述、表示和规格上的错误&#xff0c…