并发编程之Atomic原子操作类
基本类型:AtomicInteger、AtomicBoolean、AtomicLong
引用类型:AtomicReference、AtomicMarkableReference、AtomicStampedReference
数组类型:AtomicIntegerArray、AtomicLongArray、AtomicReferenceArray
对象属性原子修改器:AtomicIntegerFieldUpdater、AtomicLongFieldUpdater、AtomicReferenceFieldUpdater
原子类型累加器:DoubleAccumulator、DoubleAdder、LongAccumulator、LongAdder、Striped64
1. 基本类型
以AtomicInteger为例,它提供了原子计数器和比较交换功能
1.1 原子计数器
addAndGet()- 以原子方式将给定值添加到当前值,并在添加后返回新值。
getAndAdd() - 以原子方式将给定值添加到当前值并返回旧值。
incrementAndGet()- 以原子方式将当前值递增1并在递增后返回新值。它相当于i ++操作。
getAndIncrement() - 以原子方式递增当前值并返回旧值。它相当于++ i操作。
decrementAndGet()- 原子地将当前值减1并在减量后返回新值。它等同于i-操作。
getAndDecrement() - 以原子方式递减当前值并返回旧值。它相当于-i操作。
使用示例:
package org.example.concurrent;import lombok.extern.slf4j.Slf4j;
import org.junit.Test;import java.util.concurrent.atomic.AtomicInteger;@Slf4j
public class AtomicIntegerTest {@Testpublic void test() {AtomicInteger atomic = new AtomicInteger(0);log.debug("atomic={}", atomic);log.debug("atomic.addAndGet(1)={}, atomic={}", atomic.addAndGet(1), atomic);log.debug("atomic.getAndAdd(1)={}, atomic={}", atomic.getAndAdd(1), atomic);log.debug("atomic.incrementAndGet()={}, atomic={}", atomic.incrementAndGet(), atomic);log.debug("atomic.getAndIncrement()={}, atomic={}", atomic.getAndIncrement(), atomic);log.debug("atomic.decrementAndGet()={}, atomic={}", atomic.decrementAndGet(), atomic);log.debug("atomic.getAndDecrement()={}, atomic={}", atomic.getAndDecrement(), atomic);}
}
打印结果:
18:34:38.578 [main] DEBUG o.e.concurrent.AtomicIntegerTest - atomic=0
18:34:38.581 [main] DEBUG o.e.concurrent.AtomicIntegerTest - atomic.addAndGet(1)=1, atomic=1
18:34:38.581 [main] DEBUG o.e.concurrent.AtomicIntegerTest - atomic.getAndAdd(1)=1, atomic=2
18:34:38.581 [main] DEBUG o.e.concurrent.AtomicIntegerTest - atomic.incrementAndGet()=3, atomic=3
18:34:38.581 [main] DEBUG o.e.concurrent.AtomicIntegerTest - atomic.getAndIncrement()=3, atomic=4
18:34:38.581 [main] DEBUG o.e.concurrent.AtomicIntegerTest - atomic.decrementAndGet()=3, atomic=3
18:34:38.581 [main] DEBUG o.e.concurrent.AtomicIntegerTest - atomic.getAndDecrement()=3, atomic=2
1.2 比较和交换功能
比较和交换操作将内存中的内容与给定值进行比较,并且只有它们相同时,才将该内存位置的内容修改为给定的新值。这是作为单个原子操作完成的。
boolean compareAndSet(int expect, int update);//设置成功返回true,否则返回false
代码示例:
package org.example.concurrent;import lombok.extern.slf4j.Slf4j;
import org.junit.Test;import java.util.concurrent.atomic.AtomicInteger;@Slf4j
public class AtomicIntegerTest {@Testpublic void test() {AtomicInteger atomic = new AtomicInteger(0);log.debug("atomic={}, atomic.compareAndSet(0, 22) = {}", atomic.get(),atomic.compareAndSet(0, 22));// atomic中的值已经为22, 下面的修改会失败log.debug("atomic={}, atomic.compareAndSet(0, 22) = {}", atomic.get(),atomic.compareAndSet(0, 22));}
}
打印结果:
19:03:32.844 [main] DEBUG o.e.concurrent.AtomicIntegerTest - atomic=0, atomic.compareAndSet(0, 22) = true
19:03:32.847 [main] DEBUG o.e.concurrent.AtomicIntegerTest - atomic=22, atomic.compareAndSet(0, 22) = false
2. 引用类型
2.1 AtomicReference
原子引用可以保证你在修改引用的对象时(引用对象1改为引用对象2)的线程安全性。compareAndSet比较的是否是相同的对象,不是调用对象的equal比较。
示例:
package org.example.concurrent;import lombok.AllArgsConstructor;
import lombok.EqualsAndHashCode;
import lombok.Getter;
import lombok.extern.slf4j.Slf4j;
import org.junit.Test;import java.util.concurrent.atomic.AtomicReference;@Slf4j
public class AtomicReferenceTest {@Testpublic void test() {Person person1 = new Person("张三");Person person2 = new Person("李四");AtomicReference<Person> atomic = new AtomicReference<>(person1);log.debug("atomic={}, atomic.compareAndSet({}, {}) = {}", atomic.get(), person1, person2,atomic.compareAndSet(person1, person2));log.debug("atomic={}, atomic.compareAndSet({}, {}) = {}", atomic.get(), person1, person2,atomic.compareAndSet(person1, person2));// 创建一个新的李四对象,设置仍然失败log.debug("atomic={}, atomic.compareAndSet({}, {}) = {}", atomic.get(), person2, person1,atomic.compareAndSet(new Person("李四"), person1));}@AllArgsConstructor@Getter@EqualsAndHashCodepublic static class Person {private final String name;@Overridepublic String toString() {return getName();}}
}
打印结果:
19:01:21.260 [main] DEBUG o.e.concurrent.AtomicReferenceTest - atomic=张三, atomic.compareAndSet(张三, 李四) = true
19:01:21.263 [main] DEBUG o.e.concurrent.AtomicReferenceTest - atomic=李四, atomic.compareAndSet(张三, 李四) = false
19:01:21.263 [main] DEBUG o.e.concurrent.AtomicReferenceTest - atomic=李四, atomic.compareAndSet(李四, 张三) = false
2.2 AtomicMarkableReference 带标记的原子引用
标记只能携带true和false两个值。提供的方法有:
- getReference() 获取当前引用的对象
- isMarked() 返回标记的值
-
compareAndSet(refer, newRefer, mark, newMark),引用对象和标记都相同时,重新设置引用和标记,返回更新成功/失败
-
boolean attemptMark(ref, newMark), 引用对象相同时,更新标记,返回跟新成功/失败
代码示例:
package org.example.concurrent;import lombok.extern.slf4j.Slf4j;
import org.junit.Test;import java.util.concurrent.atomic.AtomicMarkableReference;@Slf4j
public class AtomicMarkableReferenceTest {@Testpublic void test() {AtomicMarkableReference<String> atomic = new AtomicMarkableReference<>("abc", true);log.debug("reference = {}, mark={}", atomic.getReference(), atomic.isMarked());log.debug("atomic.compareAndSet(\"abc\",\"def\", false, true) = {}",atomic.compareAndSet("abc", "def", false, true));log.debug("atomic.compareAndSet(\"abc\",\"def\", false, true) = {}",atomic.compareAndSet("abc", "def", true, true));log.debug("reference = {}, mark={}", atomic.getReference(), atomic.isMarked());log.debug("atomic.attemptMark(\"def\", false) = {}, atomicMark={}",atomic.attemptMark("def", false), atomic.isMarked());}
}
执行结果:
19:48:27.319 [main] DEBUG o.e.c.AtomicMarkableReferenceTest - reference = abc, mark=true
19:48:27.322 [main] DEBUG o.e.c.AtomicMarkableReferenceTest - atomic.compareAndSet("abc","def", false, true) = false
19:48:27.322 [main] DEBUG o.e.c.AtomicMarkableReferenceTest - atomic.compareAndSet("abc","def", false, true) = true
19:48:27.322 [main] DEBUG o.e.c.AtomicMarkableReferenceTest - reference = def, mark=true
19:48:27.322 [main] DEBUG o.e.c.AtomicMarkableReferenceTest - atomic.attemptMark("def", false) = true, atomicMark=false
2.3 AtomicStampedReference带版本号的原子引用
AtomicStampedReference 和 AtomicMarkableReference类似,只不过标记由boolean改为了int,方法attemptMark变为了attemptStamp;方法isMarked变为了getStamp
3. 数组类型
AtomicIntegerArray对数组元素的操作是原子的,提供基于数组下标的get,set,compareAndSet等操作。创建时需要提供操作的数组或者指定数组的大小。
package org.example.concurrent;import lombok.extern.slf4j.Slf4j;
import org.junit.Test;import java.util.concurrent.atomic.AtomicIntegerArray;@Slf4j
public class AtomicIntegerArrayTest {@Testpublic void test() {// 创建有10个元素的数组AtomicIntegerArray atomic = new AtomicIntegerArray(10);atomic.set(0, 1); // 第0个位置设置为1atomic.get(0); // 获取第0个位置的值。atomic.getAndAdd(2, 1); // 获取第2个位置的值,数组中的值+1atomic.compareAndSet(0, 1, 2); // 第0个位置的值如果时1的话,修改为2}
}
相关文章:
并发编程之Atomic原子操作类
基本类型:AtomicInteger、AtomicBoolean、AtomicLong 引用类型:AtomicReference、AtomicMarkableReference、AtomicStampedReference 数组类型:AtomicIntegerArray、AtomicLongArray、AtomicReferenceArray 对象属性原子修改器:…...
管家婆辉煌Ⅱ 13.32版安装方法
因管家婆辉煌版已经长期不更新,现已经出现蓝屏的问题,故此新开此贴,慢慢更新安装方法。 首先管家婆下载地址:http://www.grasp.com.cn/download.aspx?id116 先安装sql server 2008 下载后,运行安装,请注…...
常见的接口优化技巧思路
一、背景 针对老项目,去年做了许多降本增效的事情,其中发现最多的就是接口耗时过长的问题,就集中搞了一次接口性能优化。本文将给小伙伴们分享一下接口优化的通用方案。 二、接口优化方案总结 1.批处理 批量思想:批量操作数据…...
【Java EE】-使用Fiddler抓包以及HTTP的报文格式
作者:学Java的冬瓜 博客主页:☀冬瓜的主页🌙 专栏:【JavaEE】 分享: 在满园弥漫的沉静的光芒之前,一个人更容易看到时间,并看到自己的身影。——史铁生《我与地坛》 主要内容:使用FIddler抓包的…...
Java异步编程
Java异步编程 1、什么是java异步编程2、异步编程有什么作用3、异步编程常用于哪些业务4、异步编程的方式5、Async异步调用Async简介 1、什么是java异步编程 Java异步编程是一种处理并发问题的技术,它可以在执行耗时操作的同时,不阻塞主线程,…...
C++类与对象(二)——构造函数与析构函数
文章目录 一.类的默认6个成员函数二.构造函数1.引例2.构造函数的概念及特性 三.析构函数😋析构函数的特性 前言: 上篇文章初步认识了类以及类的相关知识,本篇将继续深入学习类与对象——类的默认6个成员函数: 一.类的默认6个成员函…...
c++标准模板(STL)(std::array)(四)
定义于头文件 <array> template< class T, std::size_t N > struct array;(C11 起) std::array 是封装固定大小数组的容器。 此容器是一个聚合类型,其语义等同于保有一个 C 风格数组 T[N] 作为其唯一非静态数据成员的结构体。不同于 C 风格数…...
vue3计算属性
计算属性 模板中的表达式虽然方便,但也只能用来做简单的操作。如果在模板中写太多逻辑,会让模板变得臃肿,难以维护。推荐使用计算属性来描述依赖响应式状态的复杂逻辑 基础示例 不够好的示例 模板中使用了表达式,不够直观&…...
Java 中的访问修饰符有哪些(九)
Java 中的访问修饰符用于限制类、接口、字段和方法的访问范围,它们分别表示不同的访问控制级别。Java 中共有四种访问修饰符:public、protected、default 和 private。 public public 是最开放的访问修饰符,用于指定公共访问级别。被 publi…...
HR员工管理的三重境界:管事、管人、管心
在一个公司里,员工来来往往是常态,虽说我们不能替他们决定,但是一定是与公司的管理者有一定的关系。马云曾经说过:“一个员工离职,不外乎两种原因,一是钱没给到位;二是心里委屈了”。一句话就是…...
延迟队列与SpringBoot实战
延迟队列与SpringBoot实战 概念 延时队列,队列内部是有序的,最重要的特性就体现在它的延时属性上,延时队列中的元素是希望在指定时间到了以后或之前取出和处理,简单来说,延时队列就是用来存放需要在指定时间被处理的元素的队列 …...
【算法】九键输入法
题目: 输入数字字符串, 输出这串字符对应的九键输入法有可能出现的所有情况 算法: 定义了一个全局变量 g_numStr,其中存储了每个数字对应的字母。定义了一个递归函数 str_combine,用于将每个数字对应的字母进行组合。str_combin…...
jvm之类加载器
写在前面 当我们通过javac命令将java源代码编译为Java字节码后,必须通过类加载器将其加载到jvm中才能运行,所以类加载器是jvm中非常重要的一个组成部分,本文我们就一起来看下吧! 1:类的生命周期 类的生命周期如下图…...
Chapter4:频率响应法(上)
第四章:频率响应法 Exercise4.1 已知微分网络和积分网络电路图如下图所示,求网络的频率特性。 解: 【图 ( a ) ({\rm a}) (a)微分网络】 由微分网络电路图可得:...
【6. 激光雷达接入ROS】
欢迎大家阅读2345VOR的博客【6. 激光雷达接入ROS】🥳🥳🥳 2345VOR鹏鹏主页: 已获得CSDN《嵌入式领域优质创作者》称号👻👻👻,座右铭:脚踏实地,仰望星空&#…...
Java 基础进阶篇(三)—— 面向对象的三大特征之二:继承
文章目录 一、继承概述二、内存运行原理 ★三、继承的特点四、继承后:成员变量和方法的访问特点五、继承后:方法重写六、继承后:子类构造器的特点七、继承后:子类构造器访问父类有参构造器八、this、super 总结 一、继承概述 Jav…...
[angstromctf 2023] 部分
这个比赛打了个开头就放弃了,最近放弃的比较多,国外的网太慢,国内的题太难。 Crypto ranch 这题直接给出密文这提示 rtkw{cf0bj_czbv_nvcc_y4mv_kf_kip_re0kyvi_uivjj1ex_5vw89s3r44901831} Caesar dressing is so 44 BC... 然后是加密程序…...
死信队列
死信队列 死信的概念 先从概念解释上搞清楚这个定义,死信,顾名思义就是无法被消费的消息,字面意思可以这样理解,一般来说,producer 将消息投递到 broker 或者直接到queue 里了,consumer 从 queue 取出消息…...
基于YOLOv5的目标检测系统详解(附MATLAB GUI版代码)
摘要:本文重点介绍了基于YOLOv5目标检测系统的MATLAB实现,用于智能检测物体种类并记录和保存结果,对各种物体检测结果可视化,提高目标识别的便捷性和准确性。本文详细阐述了目标检测系统的原理,并给出MATLAB的实现代码…...
使用ChatGPT工具阅读文献的实战教程
大家好,我是herosunly。985院校硕士毕业,现担任算法研究员一职,热衷于机器学习算法研究与应用。曾获得阿里云天池比赛第一名,CCF比赛第二名,科大讯飞比赛第三名。拥有多项发明专利。对机器学习和深度学习拥有自己独到的见解。曾经辅导过若干个非计算机专业的学生进入到算法…...
【国密算法C语言性能优化白皮书】:20年密码学工程师亲授SM2/SM4/SM3三级加速实战(含AVX2+国密Bouncy Castle深度适配)
第一章:国密算法C语言性能优化全景概览国密算法(SM2/SM3/SM4)作为我国商用密码体系的核心,其在嵌入式设备、金融终端与物联网节点中的高效实现,直接关系到系统吞吐量、功耗与实时性。C语言因其贴近硬件、可控性强&…...
GP2Y0A02YK0F红外测距传感器嵌入式驱动与非线性校准
1. 红外测距传感器技术解析与嵌入式实现1.1 GP2Y0A02YK0F传感器原理与特性GP2Y0A02YK0F是夏普(Sharp)公司推出的模拟输出型红外测距传感器模块,专为中距离非接触式距离检测设计。该模块采用三角测量法(Triangulation Method&#…...
清音刻墨Qwen3效果展示:看它如何实现“字字精准,秒秒不差”
清音刻墨Qwen3效果展示:看它如何实现"字字精准,秒秒不差" 1. 字幕对齐技术的革命性突破 在视频制作领域,字幕同步一直是个令人头疼的问题。传统字幕制作要么耗费大量时间手动调整,要么依赖自动语音识别(ASR)工具&…...
Pixel Dimension Fissioner环境部署:Mac M2芯片原生运行像素工坊教程
Pixel Dimension Fissioner环境部署:Mac M2芯片原生运行像素工坊教程 1. 工具介绍 Pixel Dimension Fissioner(像素语言维度裂变器)是一款基于MT5-Zero-Shot-Augment核心引擎构建的文本改写与增强工具。它将传统AI工具转化为一个充满活力的…...
从入门到精通:Kafka核心原理与实战避坑指南
在分布式系统中,消息队列是实现高可用、高并发、解耦的核心组件,而Kafka作为当下最流行的分布式消息队列之一,凭借高吞吐量、高可靠性、可扩展性等优势,广泛应用于日志收集、消息分发、流处理等场景。无论是后端开发、大数据处理&…...
PCTG-9016应用案例——E+H PROMAG 50 DP电磁流量计与ModbusTCP监控系统对接
一、项目背景与需求项目现场有一台EH品牌PROMAG 50 DP型电磁流量计,设备配置PROFIBUS DP通信接口。需要将流量计的瞬时流量和累积流量数据接入监控系统,用于数据的监控和统计。项目中控端部署的监控系统采用ModbusTCP工业通信协议,因协议类型…...
【上位机心法】别让传感器数据卡死你的 UI!撕碎 Qt/QML 渲染黑盒,用 C++ 后端打造 144Hz 零延迟工业仪表盘
摘要:当底层的 STM32 以每秒上千次的频率向电脑疯狂倾泻弹性波或高频震源数据时,如果你的 Qt 上位机界面开始卡顿、甚至假死,不要怪电脑配置低,请反思你的渲染架构。本文将无情揭露 信号与槽 (Signals and Slots) 在极高频场景下的…...
RMBG-2.0一文详解:从模型结构、推理流程到WebUI交互逻辑全梳理
RMBG-2.0一文详解:从模型结构、推理流程到WebUI交互逻辑全梳理 1. 背景去除新选择:为什么RMBG-2.0值得关注 在图像处理领域,背景去除一直是个高频需求。无论是电商商品图处理、证件照制作,还是短视频内容创作,都需要…...
Python实战:用nltk库5步搭建你的第一个n-gram文本生成器(附古诗生成案例)
Python实战:用nltk库5步搭建你的第一个n-gram文本生成器(附古诗生成案例) 在自然语言处理领域,文本生成一直是个令人着迷的话题。想象一下,计算机能够模仿人类写作风格创作出连贯的文字,这背后离不开语言模…...
别再混淆了!一文讲清NTLMv1、NTLMv2哈希的区别与各自的破解方法(附Hashcat/John命令)
深入解析NTLMv1与NTLMv2哈希:从原理到实战破解 在Windows网络认证体系中,NTLM协议作为经典的身份验证机制,至今仍广泛应用于企业内网环境。许多安全从业者在渗透测试或安全评估过程中,常会遇到需要破解NTLM哈希的情况。然而&#…...
