多线程(JUC, ReentrantLock, 原子类, 线程池, 信号量 Semaphore, CountDownLatch)

news2024/11/15 19:49:56

JUC

Java.util.concurrent 包, 存放了并发编程相关的组件, 目的是更好的支持高并发任务 (多线程只是实现并发编程的一种具体方式 …)


ReentrantLock

可重入互斥锁, 和 synchronized 定位类似, 用来实现互斥效果, 保证线程安全.

  • synchronized 对对象加锁, 保护临界资源
  • ReentreatLock 使用 lock 方法和 unlock 方法,加锁对象是 ReentrantLock 的实例

在这里插入图片描述

核心方法

  • lock(): 加锁, 获取不到锁就死等
  • trylock(超时时间):尝试加锁, 如果获取不到锁, 等待一段时间后就放弃加锁
  • unlock(): 解锁

ReentrantLock 使用

由于 reentreatLock 需要手动释放, 因此推荐 try finally 的写法

public class ReentreatLockDemo {
    public static void main(String[] args) {
        ReentrantLock reentrantLock = new ReentrantLock();
//        reentrantLock.lock();
        boolean ok = reentrantLock.tryLock();
        try {
            if(ok) {
                // do
            } else {
                // undo
            }
        } finally {
            reentrantLock.unlock();
        }
    }
}

ReentrantLock 和 synchronized 比较

  1. synchronized 是关键字, 是 JVM 内部实现的
    ReentrantLock 是标准库的一个类, 在 JVM 外实现 (基于 Java 实现)
  1. synchronized 是非公平锁
    ReentrantLock 默认是非公平锁, 但是提供了公平锁版本的实现
    ReentrantLock reentrantLock = new ReentrantLock(true);
  1. ReentrantLock 提供更灵活的加锁方式:
    ReentrantLock reentrantLock = new ReentrantLock(true);
    reentrantLock.tryLock();
  1. ReentrantLock 提供更强大, 更方便的等待通知机制
    synchronized 搭配 wait() notify()使用, notify() 是随机唤醒等待队列的线程
    ReentrantLock 搭配 Condition 类. 可以唤醒指定的线程

原子类

原子类内部用的是 CAS 实现, 更高效的解决了线程安全问题
原子类提供了线程安全的自增自减等操作


原子类有以下几种 :
在这里插入图片描述


原子类的常见方法 (以 AtomicInteger 为例)

在这里插入图片描述

public class Main {
    public static void main(String[] args) {
        AtomicInteger atomicInteger = new AtomicInteger();
        Scanner scanner = new Scanner(System.in);
        int x = scanner.nextInt();

        atomicInteger.getAndIncrement();// i++;
        atomicInteger.incrementAndGet();// ++i;
        atomicInteger.getAndDecrement();// i--;
        atomicInteger.decrementAndGet();// --i;
        atomicInteger.addAndGet(x);     // i+=x;
        atomicInteger.get();            // x
    }
}

线程池

之前写过, 挂个链接这里不再复制粘贴了 — https://editor.csdn.net/md/?articleId=136715895


信号量 Semaphore

信号量表示 "可用资源的个数" .本质上是一个计数器

Semaphore 提供了 P,V 操作
P 操作: 申请一个可用资源, 计数器 - 1
V 操作: 释放一个可用资源, 计数器 + 1

当可用资源个数为 0 时, 再进行 P 操作, 就会出现阻塞等待清空 (资源为零, 无法继续消耗了), 直到有线程让信号量大于零, 才会唤醒该阻塞的线程

锁 可可以视为计数器为 1 的信号量, 二元信号量

  • 锁是信号量的一种特殊情况
  • 信号量是锁的一般表达

总结: 信号量的表达含义范围更广


Semaphore 的简单使用
在这里插入图片描述


代码示例

public class Main {
    public static void main(String[] args) {
        // 参数是可用资源的个数(信号量的初始值)
        Semaphore semaphore = new Semaphore(4);


        for (int i=0;i<20;i++) {
            Thread t = new Thread(() -> {
                try {
                    System.out.println("申请资源");
                    semaphore.acquire();

                    System.out.println("持有资源");
                    Thread.sleep(1000);

                    System.out.println("释放资源");
                    semaphore.release();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }

            });
            t.start();
        }
    }
}

运行结果
在这里插入图片描述

有兴趣可以仔细看看运行结果, 同一时刻最多只有 4个线程能够持有锁, 这就是信号量的存在意义


CountDownLatch

同时等待 N 个任务执行结束 (和 join() 功能类似)


在这里插入图片描述

核心API

  • await(): 阻塞等待线程, 直至任务全部完成
  • getCount(): 获取剩余未完成任务个数
  • countDown(): 未完成任务个数 -1

代码示例

public class CountDownLatchDemo {
    public static void main(String[] args) throws InterruptedException {
        // 参数代表需要等待的任务数量
        CountDownLatch countDownLatch = new CountDownLatch(5);

        for (int i = 0; i < 5; i++) {
            Thread t = new Thread(() -> {
                System.out.println("完成一个任务");
                // countDown() 方法, 代表完成一个任务
                countDownLatch.countDown();
            });

            t.start();
            Thread.sleep(1000);
        }
        // await()方法, 用于阻塞线程
        // 直至 countDownLatch 内任务全部完成, 才会往下继续走
        countDownLatch.await();
        System.out.println("任务全部完成");
    }
}

运行结果
在这里插入图片描述

运行过代码会发现, 每间隔一秒输出一次 “完成一个任务”, 5秒之后输出 “任务全部完成”

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

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

相关文章

吐槽FineDataLink工具Format函数处理日期转字符串格式的说明文档

一.背景 为公司师带徒的任务做些记录。 二.文档存在的问题 1.文档情况 FORMAT-格式转换- FineBI帮助文档 FineBI帮助文档 函数定义&#xff1a; FORMAT(object,format) formart的格式有哪些呢&#xff1f;我们截图看看&#xff1a; 2.文档说明不足问题 同事的需求是把时…

【C++ STL】string类最全解析(什么是string?string类的常用接口有哪些?)

目录 一、前言 二、什么是 string ? &#x1f4a6; string 类的基本概念 &#x1f4a6; string 类与 char * 的区别 &#x1f4a6; string 类的作用 &#x1f4a6; 总结 三、string 的常用接口详解 &#x1f4a6;string 类对象的默认成员函数 ① 构造函数(初始化) ② 赋值…

无管理员权限linux系统安装cmake

去官网下载所需版本&#xff1a;https://cmake.org/download/ 之后解压 安装 cd cmake-x.xx.xx ./bootstrap DESTDIRpath make install makebashrc中添加 export PATH"path/bin:$PATH"查看是否安装成功 source ~/.bashrc cmake --version安装成功 参考&#x…

关于汽车中网改装需要报备吗?(第二天)

车联网改造需要申报吗&#xff1f; 今天2022年10月20日&#xff0c;小编就给大家介绍一下车联网改装是否需要申报的相关知识。 让我们来看看。 汽车格栅改装无需申报。 这种年检可以直接通过。 您不必担心&#xff0c;因为汽车格栅对于实车的外观来说并不陌生&#xff0c;因此…

若依添加页面

背景&#xff1a;我想增加的是一个收支管理的页面 views中直接添加income文件夹&#xff0c;里面放着index.vue 网页的菜单中添加这个页面的菜单

基于springboot的医院医护人员排班系统

技术&#xff1a;springbootvuemysql 一、系统背景 本医护人员排班系统管理员&#xff0c;医护。管理员功能有个人中心&#xff0c;医院信息管理&#xff0c;医护信息管理&#xff0c;医护类型管理&#xff0c;排班信息管理&#xff0c;排班类型管理&#xff0c;科室信息管理&…

中国网怎么投稿,发稿流程,中国网发稿需要多少钱?(附中国网各频道价格明细)

中国网是一家专业的新闻媒体平台&#xff0c;拥有庞大的读者群体和高质量的新闻内容。很多企业和个人都希望能够在中国网上发布自己的文章或新闻&#xff0c;以增加曝光度和影响力。那么&#xff0c;中国网如何投稿&#xff1f;发稿流程是怎样的&#xff1f;又需要多少费用呢&a…

4款原创检测工具,帮你快速掌握文章质量

在如今信息爆炸的时代&#xff0c;人们在网上发布和浏览大量的文字内容。然而&#xff0c;随之而来的问题是大量的抄袭和非原创内容充斥着网络空间&#xff0c;这不仅影响了文章的质量&#xff0c;也降低了读者对内容的信任度。为了帮助作者和编辑快速提升文章质量&#xff0c;…

XCode升级错误:Command CompileC failed with a nonzero exit code 解决办法

升级完XCode之后&#xff0c;bulid失败&#xff0c;出现如下错误&#xff1a; 问题1&#xff1a; xcrun: error: invalid active developer path (/Library/Developer/CommandLineTools), missing xcrun at: /Library/Developer/CommandLineTools/usr/bin/xcrunCommand Compi…

一命通关广度优先遍历

前言 在这篇文章之前&#xff0c;已对非线性结构遍历的另一种方法——深度优先遍历进行了讲解&#xff0c;其中很多概念词都是共用的。为了更好的阅读体验&#xff0c;最好先在掌握或起码了解dfs的基础上&#xff0c;再来阅读本文章&#xff0c;否则因为会有很多概念词看不明白…

Spring框架(下半部分 -AOP)

1.2 AOP相关的概念 1.2.1 AOP的概述 什么是AOP的技术&#xff1f; 在软件业&#xff0c;AOP为Aspect Oriented Programming的缩写&#xff0c;意为&#xff1a;面向切面编程 AOP是一种编程范式&#xff0c;隶属于软工范畴&#xff0c;指导开发者如何组织程序结构AOP最早由AO…

【开源-土拨鼠充电系统】鸿蒙 HarmonyOS 4.0 App+微信小程序+云平台

✨本人自己开发的开源项目&#xff1a;土拨鼠充电系统 ✨踩坑不易&#xff0c;还希望各位大佬支持一下&#xff0c;在Gitee或GitHub给我点个 Start ⭐⭐&#x1f44d;&#x1f44d; ✍Gitee开源项目地址&#x1f449;&#xff1a;https://gitee.com/cheinlu/groundhog-charging…

多维时序 | MATLAB实现BiTCN-selfAttention自注意力机制结合双向时间卷积神经网络多变量时间序列预测

多维时序 | MATLAB实现BiTCN-selfAttention自注意力机制结合双向时间卷积神经网络多变量时间序列预测 目录 多维时序 | MATLAB实现BiTCN-selfAttention自注意力机制结合双向时间卷积神经网络多变量时间序列预测预测效果基本介绍模型描述程序设计参考资料 预测效果 基本介绍 1.M…

深度学习_20_卷积中的填充与步幅

如果图片本身比较小&#xff0c;卷积之后输出也会很小&#xff0c;那么可以在图片与卷积核相乘之前先填充一下&#xff0c;让输出为预期大小 一般填充后输入&#xff0c;输出相同 当图片比较大的时候&#xff0c;如果利用卷积核去得到我们想要的大小的话&#xff0c;得用到多层…

Linux虚拟主机如何快速卸载一键安装的网站程序

接到一位用户发送的请求帮助&#xff0c;想要卸载通过Softaculous一键安装的网站程序&#xff0c;但是没有找到地方&#xff0c;根据与该用户的沟通得知是一个小白用户&#xff0c;想要自己搭建一个博客类的网站&#xff0c;了解到Hostease 有适合新用户并且带管理面板的产品Li…

Nginx:部署及配置详解(linux)

Nginx&#xff1a;部署及配置详解&#xff08;linux&#xff09; 1、nginx简介2、安装编译工具及库文件3、安装 pcre4、nginx安装5、nginx配置文件nginx.conf组成6、nginx配置实例-反向代理7、nginx 配置实例-负载均衡 &#x1f496;The Begin&#x1f496;点点关注&#xff0c…

代码随想录算法训练营第day53|1143.最长公共子序列 、 1035.不相交的线、 53. 最大子序和 动态规划

目录 1143.最长公共子序列 1035.不相交的线 53. 最大子序和 1143.最长公共子序列 力扣题目链接(opens new window) 给定两个字符串 text1 和 text2&#xff0c;返回这两个字符串的最长公共子序列的长度。 一个字符串的 子序列 是指这样一个新的字符串&#xff1a;它是由原…

【嵌入式——QT】QT Charts

【嵌入式——QT】QT Charts 概述Qt提供的坐标轴类QChartQLineSeriesQValueAxis常见图表及用到的序列类图示代码示例 概述 QT Charts模块是一组易于使用的图表组件&#xff0c;它基于Qt的Graphics View架构&#xff0c;其核心组件是QChartView和QChart&#xff0c;QChartView父…

【傻瓜文档】鼎利测试软件Pilot Pioneer-② 数据分析与处理功能

数据分析与处理功能 数据导入、解码 ① 导入数据 两种方式: 添加新数据导入历史数据不需要的数据可以移除 ② 数据解码 正式格式的数据可以重解码,例如Rcu、dcf等等,尽量选择正式数据。 两种方式: 选中数据,右键选择重解码双击信令重解码后的文件会生成为DitIB的文件…

第二节 动态面板使用实例操作

1、使用元件库 2、使用添加元件库 1.添加外部元件库 2.自做元件库 保存为*.rplib文件&#xff0c;就可以给其他项目使用了。 3、制作母板 1、新建模版 2、导入旧模版 此操作主要用于想使用原型中已经做好母版&#xff0c;有时直接粘贴又粘贴不成功&#xff0c;需要按下面方式…