JavaEE—— Callable接口、JUC的常见类、线程按安全的集合类(八股)

news2025/1/16 1:36:58

文章目录

  • 一、Callable 接口
  • 二、JUC的常见类
    • 1. ReentrantLock
    • 2. 原子类(简单知晓)
    • 3.信号量 Semaphore
    • 4.CountDownLatch(简单了解)
  • 三、线程安全的集合类
    • 1.多线程环境使用 ArrayList
    • 2.多线程使用哈希表

一、Callable 接口

Callable 接口类似于 Runnable 接口
Runnable 接口用来描述一个任务。这个任务没有返回值
Callable 也是用来描述一个任务,描述的任务有返回值

如果要使用一个线程单独计算出某个结果,此时使用 Callable 比较合适。

Callable 的使用方式:
在这里插入图片描述
这里不能直接引用到线程中,如图:
在这里插入图片描述
要将 Callable 传入到 Thread 的构造方法中,这里需要加上一层辅助类。如图:
在这里插入图片描述
在外部套上一个未来任务的泛型类就可以传入到Thread中。
在这里插入图片描述

二、JUC的常见类

java.util.concurrent 简称为 JUC 其中含有多种并发编程(多线程)的相关组件。

1. ReentrantLock

这个锁和 synchronized 锁十分相似,都是可重入互斥锁。
这里的 ReentrantLock 使用起来更加传统,使用的是 lock 和 unlock 方法进行加锁解锁

synchronized 是基于代码块的方式来加解锁的

使用方法如图:
在这里插入图片描述
这样的写法内部有着很大的问题,有可能程序员最后忘记 unlock ,如果代码中存在 return 或者异常就执行不到 unlock。这都是影响代码安全的问题。

虽然这个锁有着很多的劣势,但是仍然还是可以克服的,除此之外,它还有下面的几点优势:

  • synchronized 是非公平锁, ReentrantLock 默认是非公平锁. 可以通过构造方法传入一个 true 开启公平锁模式
  • 对于 synchronized 来讲,提供的加锁操作就是 “死等”
    但是 ReentrantLock 提供了更加灵活的等待方式:tryLock,有下面两种版本。

在这里插入图片描述

2. 原子类(简单知晓)

在这里插入图片描述

3.信号量 Semaphore

信号量,用来表示 “可用的资源个数”。本质上是一个计数器。

  • P操作: 申请一个可用资源,计数器 -1.
  • V操作: 释放一个可用资源,计数器+1.

当 P 操作时计数器为 0 ,继续 P 操作,就会出现阻塞等待的情况。

在这里设想一个停车场的场景简单解释一下,如图:

在这里插入图片描述
在这里插入图片描述

4.CountDownLatch(简单了解)

这个关键字表述的是,同时等待 N 个任务执行结束。

为了更好的理解这个关键字,这里在设想一个场景。
假设,有一个 4 人的跑步比赛。
在这里插入图片描述

  • 这里跑步开始的时间是明确的(发令枪)
  • 结束时间不明确(最后一个选手冲过重点线)

这里主要使用的方法有下面两个:

  1. await(): 主线程中使用 latch.await(); 阻塞等待所有任务执行完毕. 相当于计数器为 0 了.

需要注意的是,这里的 await中(wait 是等待,a => all 表示全部的意思)。
但是,在后面见到的很多术语中,a前缀的,大都表示 “异步”
解释同步,异步:

在这里插入图片描述

  1. latch.countDown(): 在 CountDownLatch 内部的计数器同时自减.

四个选手同时比赛,await就会阻塞前三次调用 countDown ,await 没有任何响应。当第四次调用 countDown ,await 就会被唤醒(解除阻塞)。此时认为比赛结束了。

三、线程安全的集合类

1.多线程环境使用 ArrayList

  1. 自己加锁,自己使用 synchronized 或者 ReentrantLock
  2. Collections.synchronizedList 这个会提供一些 ArrayList 的相关方法。同时是带锁的。使用方法如下:

Collections.synchronizedList(new ArrayList)
synchronizedList 的关键操作上都带有 synchronized

  1. CopyOnWriteArrayList,这个关键字也被称为 COW,也叫做 “写时拷贝”
  • 当针对这个 ArrayList 进行 操作,此时不做任何额外的工作。
  • 如果进行 操作,则要拷贝一份新的 ArrayList,针对新的进行修改,此时,如果有读操作,就先继续读 的数据。当修改完毕,此时新的数据替换旧的数据。

上面的写时拷贝有着很明显的优缺点:
优点: 在修改时不需要进行加锁。
缺点: 比较耗费空间,要求这个 ArrayList 不能太大。

这种方法可以运用到像,服务器的配置,维护,数据库的更新迭代等场景。

2.多线程使用哈希表

在这里,HashMap 是线程不安全的,HashTable 是线程安全的因为这里给关键方法添加了 synchronized。

这里更推荐使用的是 ConcurrentHashMap 这需要思考一下,ConcurrentHashMap 进行了那些优化,不 HashTable 好在哪里?两者之间有什么区别?

  1. 优化之处: ConcurrentHashMap 相比于 HashTable 大大缩小了锁冲突的概率。将一个大锁锁全部,改变成许多小锁共同操作
    这里我大概展示一下 HashTable 的逻辑形式,如图:
    在这里插入图片描述

HashTable 的做法是直接在方法上添加 synchronized,就是给整体进行了加锁,只要操作哈希表上的任何数据都会进行加锁,也就会发生锁冲突。

情况1: 如上图所示,1,2 元素在同一个链表上。此时 A 线程修改元素1,B线程修改线程2 此时会存在线程问题,就需要进行加锁处理。
情况2: 此时,如果线程A修改元素3,线程B修改元素4,因为不在一个链表上这种情况就不需要加锁。

根据上述情况,不使用 HashTable 原因就很明确了,HashTable 的锁冲突率太大,任何的两个元素都会冲突即使在不同的两个链表

ConcurrentHashMap 的逻辑形式,如图:
在这里插入图片描述
此时,锁的粒度减小了,面对 1,2 这样的情况会出现锁竞争,面对 3,4 这样的情况就不会竞争,这样就提高了代码的效率。

  1. ConcurrentHashMap 有一个相对激进的操作,针对读操作不进行加锁,只针对写操作加锁
    读和读,之间没有冲突。
    写和写,之间存在冲突。
    读和写,之间没有冲突。
    但是要注意的是,这里的读和写虽然没有冲突,但是,如果读写之间不进行加锁操作,就有可能读到一个写了一半的元素,产生脏读。要处理这样的问题,还有一个办法就是将写操作设定为原子性的(使用 volatile 关键字)

  2. 对于扩容,ConcurrentHashMap 使用了 “化整为零” 的方式。
    HashTable 扩容:
    创建一个更大的数组空间,将旧的数组上链表的元素搬运到新的数组上。此时当元素数量较多时,这样的搬运操作比较耗时!
    ConcurrentHashMap 扩容方式:
    这里采取的是每次搬运一小部分的方式。创建新的数组,旧的数组也保留,
    每次 put 时,都在新的数组上添加,同时执行一部分搬运。
    每次 get 时,旧数组和新数组都进行查询。
    每次 remove 时,只是删除元素即可。
    经过一段时间后,完全搬运完毕,释放就数组即可。

  3. ConcurrentHashMap 内部使用了 CAS 通过这个办法来进一步削减加锁操作的数目。例如:维护元素的个数。

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

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

相关文章

当阿里面试官问什么是hash?什么是布隆过滤器?什么是一致性hash?看这一篇就够了,很肝!也很干!

算法拾遗三十六hash 哈希函数特点hash表设计布隆过滤器布隆过滤器三大公式最终求解公式 一致性哈希经典数据存储经典hash缺点及解决方案虚拟节点 哈希函数特点 输入:任意长度字符串(输入域无穷大) 输出:相对有限 哈希函数无任何随…

前端,js , Error in created hook: TypeError ,有bug了

怎么兄弟,遇到bug了???你开心吗,哈哈哈哈

Linux操作系统1-命令篇

不同领域的主流操作系统 桌面操作系统 Windos Mac os Linux服务器操作系统 Unix Linux(免费、稳定、占有率高) Windows Server移动设备操作系统 Android(基于Linux,开源) ios嵌入式操作系统 Linux(机顶盒、路由器、交换机) Linux 特点:免费、开源、多用户、多任务…

【C/C++】#include<xxx.h>和#include“xxx.h“之间的区别以及寻找gcc和g++的系统头文件目录和系统库文件目录的方法

2023年7月29日&#xff0c;周六晚上 今天下午和晚上花了不少时间去研究这个C/C的头文件以及#include<xxx.h>和#include"xxx.h"之间的区别&#xff0c;收获到了很多的很有用的知识。非常值得花时间来以博客的形式总结这些学习成果。 说实话&#xff0c;我挺想…

python美化图形化界面设计,pythontkinter界面美化

大家好&#xff0c;本文将围绕python美化图形化界面设计展开说明&#xff0c;pythontkinter界面美化是一个很多人都想弄明白的事情&#xff0c;想搞清楚python美化输出模块需要先了解以下几个事情。 1、python如何做界面 PyQt&#xff0c;一个基于Qt的Python接口包&#xff0c…

网络框架重构之路plain2.0(c++23 without module) 综述

原因 plain1.1rc的不足 &#xff11;、命名空间问题 如果看过或者接触过plain的朋友&#xff0c;不难发现命名空间都是以pf_*开头。说起这个的时候&#xff0c;还是要从plain的前身plainserver&#xff08;GitHub - viticm/plainserver: A plain server, simple but stronger.&…

SpringBoot超级详解

1.父工程的父工程 在父工程的父工程中的核心依赖&#xff0c;专门用来版本管理的 版本管理。 2.父工程 资源过滤问题&#xff0c;都帮解决了&#xff0c;什么配置文件&#xff0c;都已经配置好了&#xff0c;资源过滤问题是帮助&#xff0c;过滤解决让静态资源文件能够过滤到…

HCIP OSPF链路状态类型总结

OSPF的LSA OSPF是典型的链路状态路由协议&#xff0c;使用LAS&#xff08;链路状态通告&#xff09;来承载链路状态信息。LSA是OSPF的一个核心内容&#xff0c;如果没有LSA&#xff0c;OSPF 是无法描述网络的拓扑结构及网段信息的&#xff0c;也无法传递路由信息&#xff0c;更…

跟老齐学python:数据分析,python数据分析开源软件

大家好&#xff0c;小编来为大家解答以下问题&#xff0c;python数据分析安装什么软件&#xff0c;python数据分析模块如何用&#xff0c;现在让我们一起来看看吧&#xff01;

百题千解计划【CSDN每日一练】收件邮箱(使用Python、Java、JavaScript解决)无敌的Python正则表达式、零宽负向断言

天真的人,不代表没有见过世界的黑暗,恰恰因为见到过,才知道天真的好。———三毛 🎯作者主页: 追光者♂🔥 🌸个人简介: 💖[1] 计算机专业硕士研究生💖 🌟[2] 2022年度博客之星人工智能领域TOP4🌟 🏅[3] 阿里云社区特邀专家博主🏅 🏆…

25.10 matlab里面的10中优化方法介绍—— 函数fmincon(matlab程序)

1.简述 关于非线性规划 非线性规划问题是指目标函数或者约束条件中包含非线性函数的规划问题。 前面我们学到的线性规划更多的是理想状况或者说只有在习题中&#xff0c;为了便于我们理解&#xff0c;引导我们进入规划模型的一种情况。相比之下&#xff0c;非线性规划会更加贴近…

开发集成工具pre-commit详解介绍

文章目录 pre-commit简介安装使用安装pre-commit脚本 pre-commit 官网-使用介绍 官网-可使用的钩子列表 简介 pre-commit&#xff1a;帮助你提高代码质量的工具 pre-commit是一个帮助开发人员提高代码质量的工具。它通过在提交代码之前运行一系列检查来实现这一点。这些检…

【并发编程】ForkJoinPool工作原理分析

目录 前置内容课程内容一、由一道算法题引发的思考1.算法题2.什么是归并排序法 二、什么是Fork/Join框架1.基本介绍2.ForkJoinPool2.ForkJoinPool构造函数及参数解读3.任务提交方式4.工作原理图5.工作窃取6.和普通线程池之间的区别7.ForkJoinTask 学习总结 前置内容 Q1&#x…

数学建模学习(7):Matlab绘图

一、二维图像绘制 1.绘制曲线图 最基础的二维图形绘制方法&#xff1a;plot -plot命令自动打开一个图形窗口Figure&#xff1b; 用直线连接相邻两数据点来绘制图形 -根据图形坐标大小自动缩扩坐标轴&#xff0c;将数据标尺及单位标注自动加到两个坐标轴上&#xff0c;可自定…

类和对象|六个默认成员函数|const成员函数|运算符重载

文章目录 默认成员构造函数1. 构造函数1.1 概念1.2 特性 2. 析构函数2.1 概念2.2 特性 3. 拷贝构造函数3.1 概念3.2 特性 4. 运算符重载4.1 赋值重载4.2 自增自减重载4.3 取地址操作符重载 5. const成员函数6. 取地址重载 默认成员构造函数 上一节我们说过&#xff0c;空类的大…

行为型:发布订阅模式

定义   发布订阅模式是基于一个事件&#xff08;主题&#xff09;通道&#xff0c;希望接收通知的对象Subscriber&#xff08;订阅者&#xff09;通过自定义事件订阅主题&#xff0c;被激活事件的对象 Publisher &#xff08;发布者&#xff09;通过发布主题事件的方式通知订…

STM32F103利用CubeMX配置开启定时中断

1、外部晶振8MHz&#xff0c;下载方式SWD模式&#xff0c;需求配置定时器1&#xff0c;产生每100ms一次中断 新建工程、配置晶振、选择下载方式等略 2、查阅资料&#xff0c;STM32F103的时钟树分配 3、配置CubeMX的时钟树 4、配置定时器-开启定时中断 5、配置定时时间 &…

VoxPoser:使用大语言模型(GPT-4)来对机器人操作的可组合三维值图【论文解读】

这是最近斯坦福的李飞飞团队的一篇论文:VoxPoser: Composable 3D Value Maps for Robotic Manipulation with Language Models 主要是通过大语言模型LLM和视觉语言模型VLM结合&#xff0c;来对机器人做各种日常操作&#xff0c;我们可以先来看下实际效果&#xff1a;大语言模型…

使用LangChain构建问答聊天机器人案例实战(一)

使用LangChain构建问答聊天机器人案例实战 现场演示GPT-4代码生成 本节我们会通过一个综合案例,跟大家讲解LangChain,这个案例产生的代码会直接在浏览器中运行,并且会输出结果,如图14-1所示,用户问:“What was the highest close price of IBM?”(“IBM的最高收盘价是…

【Linux命令200例】mdel删除指定目录下的多个文件

&#x1f3c6;作者简介&#xff0c;黑夜开发者&#xff0c;全栈领域新星创作者✌&#xff0c;2023年6月csdn上海赛道top4。 &#x1f3c6;本文已收录于专栏&#xff1a;Linux命令大全。 &#x1f3c6;本专栏我们会通过具体的系统的命令讲解加上鲜活的实操案例对各个命令进行深入…