并发编程学习(十一):原子数组、
1、数组类型的原子类
原子数组类型,这个其实和AtomicInteger
等类似,只不过在修改时需要指明数组下标。
CAS是按照==
来根据地址进行比较。数组比较地址,肯定是不行的,只能比较下标元素。而比较下标元素,就和元素的类型有关系了。
在java.util.concurrent.atomic
中,原子类型数组有以下四种:
类名 | 说明 |
---|---|
AtomicIntegerArray | 提供对int[]数组元素的原子性更新操作 |
AtomicLongArray | 提供对long[]数组元素的原子性更新操作 |
AtomicReferenceArray | 提供对引用类型[] 数组元素的原子性更新操作 |
AtomicBooleanArray | 原子更新布尔类型数组的元素 |
使用原子的方式更新数组里的某个元素。
2、常用方法
AtomicIntegerArray
和AtomicLongArray
的使用方式差别不大,AtomicReferenceArray
因为他的参数为引用数组,所以跟前两个的使用方式有所不同。AtomicBooleanArray
在生产中使用的很少。
本次只对AtomicLongArray
和AtomicReferenceArray
方法进行详细的介绍。
2.1、AtomicLongArray介绍
2.1.1、构造方法
方法名 | 说明 |
---|---|
AtomicLongArray(int length) | 创建给定长度的新 AtomicLongArray |
AtomicLongArray(long[] array) | 创建与给定数组具有相同长度的新 AtomicLongArray ,并从给定数组复制其所有元素 |
2.1.2、方法
方法 | 说明 |
---|---|
long getAndIncrement(int i) | 以原子方式将索引 i 的元素自增 1 ,并返回旧值 |
long incrementAndGet(int i) | 以原子方式将索引 i 的元素自增 1 ,并返回减少之后的值 |
long getAndDecrement(int i) | 以原子形式将索引i 处的元素原子自减1 ,并返回旧值 |
long decrementAndGet(int i) | 以原子形式将索引i 处的元素原子自减1 ,并返回减少之后的值 |
long addAndGet(int i, long delta) | 以原子形式将给定元素与数组中索引i 的元素相加 |
long getAndSet(int i, long newValue) | 将地位i 处的元素原子设置为给定值,并返回旧值 |
long get(int i) | 获取位置 i 的当前值 |
void lazySet(int i, long newValue) | 最终将位置 i 的元素设置为给定值 |
int length() | 返回数组的长度 |
void set(int i, long newValue) | 将位置 i 的元素设置为给定值 |
boolean compareAndSet(int i,int expect,int update) | 如果当前值 == 预期值,则以原子方式将该值设置为给定的更新值。 |
boolean weakCompareAndSet(int i, int expect, long update) | 如果当前值 == 预期值,则以原子方式将位置 i 的元素设置为给定的更新值 |
long getAndUpdate(int i, LongUnaryOperator updateFunction) | 应用将给定函数利用以后值和给定值的原子更新以后值,返回旧值 |
long updateAndGet(int i, LongUnaryOperator updateFunction) | 应用将给定函数利用以后值和给定值的原子更新以后值,返回新值 |
2.1.3、示例:
package com.example.test.java.juc;import lombok.extern.slf4j.Slf4j;import java.util.concurrent.atomic.AtomicLongArray;
import java.util.function.LongBinaryOperator;
import java.util.function.LongUnaryOperator;@Slf4j(topic = "com.text.AtomicLongArray")
public class AtomicLongArrayTest {// 初始化 数组长度为 5private static AtomicLongArray arr = new AtomicLongArray(5);private static LongUnaryOperator longUnaryOperator = new LongUnaryOperator() {@Overridepublic long applyAsLong(long operand) {return operand + 10;}};private static LongBinaryOperator accumulatorFunction = new LongBinaryOperator() {@Overridepublic long applyAsLong(long left, long right) {return left + right;}};public static void main(String[] args) {for (int i = 0; i < arr.length(); i++) {log.debug("i-" + i + "=" + arr.get(i));}// 以原子形式给以后索引下标为(0)值加1,返回新值 (i++): 0log.debug("索引 0 incrementAndGet=" + arr.getAndIncrement(0));// 以原子形式给以后索引下标为(0)值加1,,返回新值(++i): 2log.debug("索引 0 incrementAndGet=" + arr.incrementAndGet(0));//以原子形式给以后索引下标为(0)值缩小 1,返回旧值 (i--):2log.debug("索引 0 incrementAndGet=" + arr.getAndDecrement(0));//以原子形式给以后索引下标为(0)值缩小 1,返回旧值 (--i):0log.debug("索引 0 incrementAndGet=" + arr.decrementAndGet(0));// 以原子形式将输出的数值与实例中的值(AtomicLongArray(0)里的value)相加,并返回后果 : 100log.debug("索引 0 addAndGet=" + arr.addAndGet(0, 100));// 获取 AtomicLongArray 的 value 100log.debug("索引 0 get=" + arr.get(0));log.debug("*********** JDK 1.8 ***********");//应用将给定函数利用给以后值和给定值的后果原子更新以后值,返回上一个值// 索引下标为 0 执行指定函数 后果为 100 + 10log.debug("索引 0 getAndUpdate=" + arr.updateAndGet(0, longUnaryOperator));// 索引下标为 1 执行指定函数 后果为 0 + 10log.debug("索引 1 getAndUpdate=" + arr.updateAndGet(1, longUnaryOperator));// 应用给定函数利用给指定下标和给定值的后果原子更新以后值,并返回后果 20log.debug("索引 1 accumulateAndGet=" + arr.accumulateAndGet(1, 10, accumulatorFunction));}
}
运行结果:
16:20:48.939 [main] DEBUG com.text.AtomicLongArray - i-0=0
16:20:48.945 [main] DEBUG com.text.AtomicLongArray - i-1=0
16:20:48.945 [main] DEBUG com.text.AtomicLongArray - i-2=0
16:20:48.945 [main] DEBUG com.text.AtomicLongArray - i-3=0
16:20:48.945 [main] DEBUG com.text.AtomicLongArray - i-4=0
16:20:48.945 [main] DEBUG com.text.AtomicLongArray - 索引 0 incrementAndGet=0
16:20:48.946 [main] DEBUG com.text.AtomicLongArray - 索引 0 incrementAndGet=2
16:20:48.946 [main] DEBUG com.text.AtomicLongArray - 索引 0 incrementAndGet=2
16:20:48.946 [main] DEBUG com.text.AtomicLongArray - 索引 0 incrementAndGet=0
16:20:48.946 [main] DEBUG com.text.AtomicLongArray - 索引 0 addAndGet=100
16:20:48.946 [main] DEBUG com.text.AtomicLongArray - 索引 0 get=100
16:20:48.946 [main] DEBUG com.text.AtomicLongArray - *********** JDK 1.8 ***********
16:20:48.946 [main] DEBUG com.text.AtomicLongArray - 索引 0 getAndUpdate=110
16:20:48.946 [main] DEBUG com.text.AtomicLongArray - 索引 1 getAndUpdate=10
16:20:48.947 [main] DEBUG com.text.AtomicLongArray - 索引 1 accumulateAndGet=20
2.2 AtomicReferenceArray介绍
2.2.1、构造方法
方法名 | 说明 |
---|---|
AtomicReferenceArray(E[] array) | 创建与给定数组具有相同长度的新 这个构造方法传入一个数组对象时,该数组对象必须是引用类型,int[]不可以,但是Integer[]的可以 |
AtomicReferenceArray(int length) | 创建给定长度的新 AtomicReferenceArray。 为数组的每一位设置什么值是没有要求的,类似于Map的形式 |
2.2.2、方法
方法 | 说明 |
---|---|
boolean compareAndSet(int i, E expect, E update) | 如果当前值 == 预期值,则以原子方式将位置 i 的元素设置为给定的更新值 |
E get(int i) | 获取位置 i 的当前值 |
E getAndSet(int i, E newValue) | 以原子方式将位置 i 的元素设置为给定值,并返回旧值 |
void lazySet(int i, E newValue) | 最终将位置 i 的元素设置为给定值 |
int length() | 返回该数组的长度 |
void set(int i, E newValue) | 将位置 i 的元素设置为给定值 |
boolean weakCompareAndSet(int i, E expect, E update) | 如果当前值 == 预期值,则以原子方式将位置 i 的元素设置为给定的更新值 |
2.2.3、代码示例
package com.example.test.java.juc;import java.util.concurrent.atomic.AtomicReferenceArray;public class AtomicReferenceArrayTest {public static void main(String[] args) {Long[] l = new Long[4];String[] s = new String[4];int[] i = new int[4];Integer[] in = new Integer[4];AtomicReferenceArray atomicReferenceArray = new AtomicReferenceArray(l);System.out.println(atomicReferenceArray.length());System.out.println(atomicReferenceArray.get(2));AtomicReferenceArray atomic = new AtomicReferenceArray(4);atomic.set(0,12);atomic.set(2,"Leefs");atomic.set(3,i);System.out.println(atomic.toString());}
}
运行结果:
4
null
[12, null, Leefs, [I@15327b79]
3、原子性测试
创建10个线程,每个线程分别对数组操作(自增)10000次(采用了函数式编程)。
package com.example.test.java.juc;import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.concurrent.atomic.AtomicIntegerArray;
import java.util.function.BiConsumer;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.function.Supplier;public class AtomicArrayTest {/**参数1:提供数组、可以是线程不安全数组或线程安全数组参数2:获取数组长度的方法参数3:自增方法,回传 array, index参数4:打印数组的方法supplier 提供者 无中生有 ()->结果function 函数 一个参数一个结果 (参数)->结果,BiFunction (参数1,参数2)->结果consumer 消费者 一个参数没结果 (参数)->void,BiConsumer (参数1,参数2)->void*/private static <T> void demo(Supplier<T> arraySupplier,Function<T,Integer> lengthFun,BiConsumer<T,Integer> putConsumer,Consumer<T> printConsumer) {// 创建线程列表List<Thread> threadList = new ArrayList<>();// 得到数组返回的元素T array = arraySupplier.get();// 后面的Integer是返回结果int length = lengthFun.apply(array);// 根据数组长度来进行遍历,对值进行累加for (int i = 0; i < length; i++) {// 每个线程对数组作 10000 次操作threadList.add(new Thread(() -> {for (int j = 0; j < 10000; j++) {putConsumer.accept(array, j%length);}}));}threadList.forEach(t -> t.start()); // 启动所有线程threadList.forEach(t -> {try {t.join();} catch (InterruptedException e) {e.printStackTrace();}});// 等所有线程结束printConsumer.accept(array);}public static void main(String[] args) {// 不安全数据demo(()->new int[10],(array)->array.length,(array, index) -> array[index]++,array-> System.out.println(Arrays.toString(array)));// 安全数据demo(()->new AtomicIntegerArray(10),(arr)-> arr.length(),(arr,index) -> arr.getAndIncrement(index),(arr)-> System.out.println(arr));}
}
运行结果:
[9035, 9053, 9034, 9049, 9039, 9041, 9032, 9027, 9040, 9025]
[10000, 10000, 10000, 10000, 10000, 10000, 10000, 10000, 10000, 10000]
相关文章:

并发编程学习(十一):原子数组、
1、数组类型的原子类 原子数组类型,这个其实和AtomicInteger等类似,只不过在修改时需要指明数组下标。 CAS是按照来根据地址进行比较。数组比较地址,肯定是不行的,只能比较下标元素。而比较下标元素,就和元素的…...

递归到动态规划:省去枚举行为
如果在动态规划的过程中没有枚举行为,那严格位置依赖和傻缓存的方式并没有太大区别,但是当有枚举行为的时候(一个位置依赖于多个位置),那严格位置依赖是有优化空间的,枚举行为也许可以省去,题目…...

服务(第二十一篇)mysql高级查询语句(二)
①视图表: 视图表是虚拟表,用来存储SQL语句的定义 如果视图表和原表的字段相同,是可以进行数据修改的; 如果两者的字段不通,不可以修改数据。 语法: 创建:create view 试图表名 as ... 查…...

MYSQL高可用配置(MHA)
1、什么是MHA MHA(Master High Availability)是一套优秀的MySQL高可用环境下故障切换和主从复制的软件。 MHA 的出现就是解决MySQL 单点的问题。 MySQL故障切换过程中,MHA能做到0-30秒内自动完成故障切换操作。 MHA能在故障切换的过程中最大…...

单精度浮点数与十进制数据相互转换
一、float基础: Float类型占4个字节,也就是32bit,其中最高位是符号位,2~9位是指数位,后边的23bit是数值位.如下所示 大部分数据的二进制形式都可以用科学计数法表示,即1.m*2^n这种形式,只要知道m和n,就能确定一个数值 二、小数位如何转变为二进制: 下面…...

PMP敏捷-4大价值观、12原则
宣言及4大价值观 个体及互动 胜于 流程和工具 以人为本 工作的软件 胜于 完整的文档 以价值为导向 客户合作 胜于 合同谈判 合作共赢 应对变更 胜于 遵循计划 拥抱变化 12原则 工作原则:精益、至简,实现这种原则的方式是“定期反省”。9、10、12 …...

K8S—Helm
一、Helm介绍 helm通过打包的方式,支持发布的版本管理和控制,很大程度上简化了Kubernetes应用的部署和管理。 Helm本质就是让k8s的应用管理(Deployment、Service等)可配置,能动态生成。通过动态生成K8S资源清单文件&a…...

ALSA内部函数调用流程
ALSA内部函数调用流程 一直都有这样的一个疑问 就是在linux系统中我们调用snd_pcm_open后,就不知道alsa内部是怎么运行的了 用户的pcm_open()相当于先对ASoC各个驱动模块startup(),再做hw_params()。 pcm_open()pcm->fd open("/dev/snd/pcm…...

Python正则表达式详解,保姆式教学,0基础也能掌握正则
正则作为处理字符串的一个实用工具,在Python中经常会用到,比如爬虫爬取数据时常用正则来检索字符串等等。正则表达式已经内嵌在Python中,通过导入re模块就可以使用,作为刚学Python的新手大多数都听说”正则“这个术语。 今天来给…...

ChatGPT 接入飞书教程,创建自己的聊天机器人
ChatGPT 接入飞书教程,创建自己的聊天机器人 一、飞书进入开发者平台。点击创建应用。二、打开Aircode,点击创建应用,上面输入名字,下面选择Node.js v16三、配置环境,点击Environments,创建四个变量,全部要大写本教程收集于: AIGC从入门到精通教程 首先,准备三个账号…...

JS生成随机数(多种解决方案)
JS生成随机数 概述 随机数是编程语言中的重要组成部分。在JavaScript中,生成随机数是一项简单的任务。本文将介绍生成随机数的各种方法。 Math.random() Math.random()是JavaScript中生成随机数最常见的方法。该方法返回介于0和1之间的随机数。例如,…...

文件IO 函数 静态库和动态库的创建 5.11
5.11 文件IO函数 1.数据读写 ssize_t read(int fd,void *buf,size_t count); 功能: 从fd对应的文件中 读取前count个字节的数据到buf缓冲区中 头文件: #include <unistd.h> 参数: fd :文件描述符 buf…...

考研日语-详解ている、てある、ていく、てくる用法
目录 一、ている用法 1. 表示现在状态 2. 表示持续动作 3. 表示经验或习惯 4. 表示结果或效果 二、てある用法 1. 表示已经完成的动作 2. 表示现在状态 3. 表示被动 三、ていく用法 1. 表示未来的动作 2. 表示逐渐变化的过程 四、てくる用法 1. 表示过去到现在的…...

Spring Security 6.x 系列【36】授权服务器篇之OpenID Connect 1.0
有道无术,术尚可求,有术无道,止于术。 本系列Spring Boot 版本 3.0.4 本系列Spring Security 版本 6.0.2 本系列Spring Authorization Server 版本 1.0.2 源码地址:https://gitee.com/pearl-organization/study-spring-security-demo 文章目录 1. 前言2. OpenID Connect…...

【计算机视觉 | Pytorch】timm 包的具体介绍和图像分类案例(含源代码)
一、具体介绍 timm 是一个 PyTorch 原生实现的计算机视觉模型库。它提供了预训练模型和各种网络组件,可以用于各种计算机视觉任务,例如图像分类、物体检测、语义分割等等。 timm 的特点如下: PyTorch 原生实现:timm 的实现方式…...

轻博客Plume的搭建
什么是 Plume ? Plume 是一个基于 ActivityPub 的联合博客引擎。它是用 Rust 编写的,带有 Rocket 框架,以及 Diesel 与数据库交互。前端使用 Ructe模板、WASM 和SCSS。 反向代理 假设我们实际访问地址为: https://plume.laosu.ml…...

机器人关节电机PWM
脉冲宽度调制(Pulse width modulation,PWM)技术。一种模拟控制方式 机器人关节电机的控制通常使用PWM(脉冲宽度调制)技术。PWM是一种用于控制电子设备的技术,通过控制高电平和低电平之间的时间比例,实现对电子设备的控制。在机器人关节电机中,PWM信号可以控制电机的…...

MPU6050详解(含源码)
前言:MPU6050是一款强大的六轴传感器,需要理解MPU6050首先得有IIC的基础,MPU6050 内部整合了 3 轴陀螺仪和 3 轴加速度传感器,并且含有一个第二 IIC 接口,可用于连接外部磁力传感器,内部有硬件算法支持. 1…...

Vue入门学习笔记:TodoList(三):实例中的数据、事件和方法
目录: Vue入门学习笔记:TodoList(一):HelloWorld Vue入门学习笔记:TodoList(二):挂载点、模板、实例 Vue入门学习笔记:TodoList(三)&a…...

怎么找到引发回流的JavaScript代码?
要找到引发回流的JavaScript代码,可以使用浏览器的开发者工具中的性能分析器。不同的浏览器有不同的名称和位置,例如Google Chrome的开发者工具中的性能分析器被称为Performance,Firefox的开发者工具中的性能分析器被称为Profiler。 以下是在…...

未来广告策划,转型还是淘汰?
在广告行业呆了十来年了,最近我越来越感觉到广告行业真的是一个需要与时俱进,并且应用场景非常广泛的一个专业。 而且由于这是一个需要创意能力的行业,所以对比于重复性容易被机器以及人工智能所代替的岗位行业来说,广告的可替代…...

【vscode远程开发】使用SSH远程连接服务器 「内网穿透」
文章目录 前言视频教程1、安装OpenSSH2、vscode配置ssh3. 局域网测试连接远程服务器4. 公网远程连接4.1 ubuntu安装cpolar内网穿透4.2 创建隧道映射4.3 测试公网远程连接 5. 配置固定TCP端口地址5.1 保留一个固定TCP端口地址5.2 配置固定TCP端口地址5.3 测试固定公网地址远程 转…...

七天从零实现Web框架Gee - 扩展
到这里前七天的任务已经完成,但我们可以对Gee框架进行一些扩展 补充HTTP请求方法 原作者只实现了 GET, POST 路由添加,其他的 PUT, DELETE 等标准 HTTP 方法未实现,实现方法也很简单,只需在gee.go中增加如下代码 // PUT define…...

什么是土壤水分传感器
土壤水分传感器又称土壤湿度传感器由不锈钢探针和防水探头构成,可长期埋设于土壤和堤坝内使用,对表层和深层土壤进行墒情的定点监测和在线测量。与数据采集器配合使用,可作为水分定点监测或移动测量的工具(即农田墒情检测仪&#…...

月薪17k需要什么水平?98年测试员的面试全过程…
我的情况 大概介绍一下个人情况,男,本科,三年多测试工作经验,懂python,会写脚本,会selenium,会性能,然而到今天都没有收到一份offer!从年后就开始准备简历,年…...

知了汇智:坚持发展产教融合,做好高校、人才与企业之间的桥梁
6月将正式迎来高校毕业季,大学生就业是聚焦全社会关注的头等大事。5月9日,成都知了汇智科技有限公司(以下简称“知了汇智”)组织开展“深化产教融合、聚焦人才培养”的主题座谈会议,联动高校与合作企业参加,…...

MyBatis缓存-一级缓存--二级缓存的非常详细的介绍
目录 MyBatis-缓存-提高检索效率的利器 缓存-官方文档 一级缓存 基本说明 一级缓存原理图 代码演示 修改MonsterMapperTest.java, 增加测试方法 结果 debug 一级缓存执行流程 一级缓存失效分析 关闭sqlSession会话后 , 一级缓存失效 如果执行sqlSession.clearCache(…...

macOS Ventura 13.4 RC2(22F63)发布
系统介绍 根据黑果魏叔官网提供:5 月 12 日消息,苹果今天面向开发人员,发布了 macOS Ventura 13.4 的第 2 个候选 RC 版本(内部版本号 22F63),距离上个候选版本相隔数天时间。 macOS Ventura 带来了台前调…...

【为什么可以相信一个HTTPS网站】
解决信用,仅仅有加密和解密是不够的。加密解密解决的只是传输链路的安全问题,相当于两个人说话不被窃听。可以类比成你现在生活 的世界——货币的信用,是由政府在背后支撑的;购房贷款的信用,是由银行在背后支撑的&…...

4.进阶篇
目录 一、按照测试对象划分 1.界面测试(UI测试) 界面测试的常见错误: 2.可靠性测试 3.容错性测试 4.文档测试 5.兼容性测试 6.易用性 7.安装卸载测试 8.安全性测试 9.性能测试 10.内存泄漏 二、按照是否查看代码 1.黑盒测试 2.…...