当前位置: 首页 > news >正文

Java并发基础:原子类之AtomicIntegerFieldUpdater全面解析

Java并发基础:原子类之AtomicIntegerFieldUpdater全面解析 - 程序员古德

本文概要

AtomicIntegerFieldUpdater类提供了一种高效、简洁的方式来原子性地更新对象的volatile字段,无需使用重量级的锁机制,它通过基于反射的API实现了细粒度的并发控制,提升了多线程环境下的性能表现。

AtomicIntegerFieldUpdater核心概念

AtomicIntegerFieldUpdater类是一个用于原子更新字段值的工具类,它特别适用于在并发环境中,当多个线程需要访问和修改某个对象的某个volatile整型字段时,能够保证该字段更新的原子性。

模拟一个业务场景,假设有一个在线书店,每个书籍都有一个库存数量字段,表示为int stockCount,当多个用户同时购买同一本书时,系统需要确保库存数量的减少是线程安全的,即不会出现超卖的情况。

在传统的同步方法中,可能会使用synchronized关键字或ReentrantLock来同步整个库存减少的方法,但这样做的话,每次只有一个线程能够执行减少库存的操作,其他线程必须等待,这在高并发环境下可能会导致性能瓶颈。

在这个场景中使用AtomicIntegerFieldUpdater类的decrementAndGet方法,这个方法会以原子方式将库存数量减1,并返回更新后的值,同时它是以原子地更新库存数量字段,而不需要对整个方法进行同步,多个线程可以同时尝试减少库存,但每次只有一个线程能够成功更新库存数量,其他线程会重新尝试,直到成功为止。

注意:使用AtomicIntegerFieldUpdater类时,库存数量字段必须是volatile修饰的,这样可以保证所有线程都能看到最新的值,同时由于AtomicIntegerFieldUpdater类是基于反射实现的,因此,它只能更新公共字段或具有公共setter方法的字段。

AtomicIntegerFieldUpdater使用案例

下面是一个简单的Java代码案例,演示了如何使用AtomicIntegerFieldUpdater类,如下代码:

import java.util.concurrent.atomic.AtomicIntegerFieldUpdater;  public class AtomicIntegerFieldUpdaterExample {  // 定义一个包含volatile字段的类  static class MyObject {  volatile value = 0; // 这个字段将被原子更新  }  // 创建一个AtomicIntegerFieldUpdater实例,用于更新MyObject的value字段  private static final AtomicIntegerFieldUpdater<MyObject> updater =  AtomicIntegerFieldUpdater.newUpdater(MyObject.class, "value");  public static void main(String[] args) {  // 创建一个MyObject实例  MyObject myObject = new MyObject();  // 输出初始值  System.out.println("Initial value: " + myObject.value);  // 使用AtomicIntegerFieldUpdater原子地增加value字段的值  updater.incrementAndGet(myObject);  // 输出更新后的值  System.out.println("Value after increment: " + myObject.value);  // 使用AtomicIntegerFieldUpdater原子地将value字段的值设置为100  updater.set(myObject, 100);  // 输出再次更新后的值  System.out.println("Value after setting to 100: " + myObject.value);  }  
}

输出将是:

Initial value: 0  
Value after increment: 1  
Value after setting to 100: 100

代码解释:

  1. 定义一个名为MyObject的静态内部类,该类有一个volatile字段value
  2. 创建一个AtomicIntegerFieldUpdater类的实例updater,该实例将用于原子地更新MyObject类的value字段。
  3. main方法中,创建了MyObject的实例,并使用updater原子地增加和设置value字段的值。
  4. 使用System.out.println来输出value字段的初始值、增加后的值和设置为100后的值。

AtomicIntegerFieldUpdater核心API

AtomicIntegerFieldUpdater类是Java的java.util.concurrent.atomic包中的一个工具类,这个类的主要是用于原子地更新指定对象的指定volatile字段,以下是该类中一些主要方法的含义:

  1. newUpdater(Class<T> tclass, String fieldName):
    这是一个静态方法,用于创建一个新的AtomicIntegerFieldUpdater,它能够以原子方式更新给定类的指定名称的volatile字段,tclass是字段所在类的Class对象,fieldName是要更新的字段的名称。
  2. get(T obj):
    这个方法获取指定对象的volatile字段的当前值,obj是包含要获取字段的对象。
  3. set(T obj, int newValue):
    这个方法以原子方式设置指定对象的volatile字段的值为newValueobj是包含要设置字段的对象,newValue是要设置的新值。
  4. lazySet(T obj, int newValue):
    这个方法最终将设置指定对象的volatile字段的值为newValue,但它允许之后的内存操作重排序,也就是说这个操作可能不是立即对其他线程可见的,它通常用于提高性能,但牺牲了一些一致性保证。
  5. getAndSet(T obj, int newValue):
    这个方法以原子方式设置指定对象的volatile字段的值为newValue,并返回该字段的旧值,obj是包含要设置字段的对象,newValue是要设置的新值。
  6. getAndAdd(T obj, int delta):
    这个方法以原子方式将给定值delta添加到指定对象的volatile字段的当前值,并返回更新前的值,obj是包含要添加字段的对象,delta是要添加的值。
  7. incrementAndGet(T obj):
    这个方法以原子方式将指定对象的volatile字段的当前值增加1,并返回更新后的值,obj是包含要增加字段的对象。
  8. decrementAndGet(T obj):
    这个方法以原子方式将指定对象的volatile字段的当前值减少1,并返回更新后的值,obj是包含要减少字段的对象。
  9. addAndGet(T obj, int delta):
    这个方法以原子方式将给定值delta添加到指定对象的volatile字段的当前值,并返回更新后的值,obj是包含要添加字段的对象,delta是要添加的值。
  10. compareAndSet(T obj, int expect, int update):
    这个方法以原子方式将指定对象的volatile字段的值与expect值进行比较,如果当前值等于expect值,则使用update值更新该字段,如果更新成功,则返回true,否则返回false,这个方法通常用于实现基于比较的同步机制,如自旋锁。

使用AtomicIntegerFieldUpdater时,必须确保被更新的字段是volatile修饰的,并且对于使用AtomicIntegerFieldUpdater的类是可访问的(即字段是public的,或者与AtomicIntegerFieldUpdater在同一个包中且字段是包私有的,或者通过其他方式使字段可访问),此外,字段也不能是static的。

AtomicIntegerFieldUpdater技术原理

AtomicIntegerFieldUpdater类用于对对象的某个volatile字段进行原子性更新,该类的实现原理基于Java的内存模型(JMM)和Unsafe类的底层操作。

实现原理

  1. Java内存模型(JMM):使用Java内存模型保证了多线程之间变量的可见性和原子性操作,使用volatile关键字确保了一个线程对变量的修改对其他线程是立即可见的,并且禁止了指令重排。
  2. Unsafe类:
    AtomicIntegerFieldUpdater的底层实现依赖于sun.misc.Unsafe类,该类提供了低级别的、非安全的、操作系统级别的访问方法,它可以直接访问内存、创建对象、数组等,而不受Java访问控制的限制。
  3. 反射:
    AtomicIntegerFieldUpdater使用反射来获取要更新的字段的Field对象,然后通过Unsafe类直接操作这个字段的内存地址。
  4. 原子操作:
    Unsafe类提供了一系列原子操作方法,如compareAndSwapInt,这是一个基于硬件支持的原子比较并交换(CAS)操作,CAS操作包括三个参数:一个内存位置(V)、预期原值(A)和新值(B),如果内存位置V的值与预期原值A相匹配,那么处理器会自动将该位置的值更新为新值B,否则,处理器不做任何操作,无论哪种情况,它都会在CAS指令之前返回该位置的值,这一过程是原子的,也就是说在执行过程中不会被其他线程打断。

底层算法

AtomicIntegerFieldUpdater的底层算法主要基于CAS操作来实现原子性更新,以incrementAndGet方法为例:

  1. 使用一个do-while循环来尝试更新字段的值。
  2. 在循环体内,首先使用Unsafe类的getIntVolatile方法获取当前字段的值。
  3. 计算新的值(当前值 + 1)。
  4. 使用Unsafe类的compareAndSwapInt方法尝试将字段的值从当前值更新为新值,如果成功,则退出循环并返回新值;如果失败(说明其他线程已经修改了该字段的值),则继续循环。

这种基于CAS的算法是一种无锁算法,也称为乐观锁算法,它不需要获取和释放锁,而是通过不断重试来确保更新的原子性,在高并发环境下,这种算法通常比传统的基于锁的算法具有更好的性能。

学习总结

Java并发基础:原子类之AtomicIntegerFieldUpdater全面解析 - 程序员古德

AtomicIntegerFieldUpdater类允许以原子方式更新对象的某个volatile字段,而无需使用synchronized关键字,这样做的优点在于减少了锁的竞争,提升了多线程环境下的性能,并且使用简单,只需通过反射指定字段即可。

但是,由于使用了反射,所以字段必须是可访问的,这可能会破坏封装性,并且,它只能更新volatile类型的字段,对于其他类型的字段或者非volatile字段则无能为力。

在使用AtomicIntegerFieldUpdater时,建议仅在确实需要原子性更新且性能是关键因素时使用,并且要要注意保持字段的可访问性,并确保字段是volatile类型的。

关注我,每天学习互联网编程技术 - 程序员古德

END!
END!
END!

往期回顾

精品文章

Java并发基础:concurrent Flow API全面解析

Java并发基础:CopyOnWriteArraySet全面解析

Java并发基础:ConcurrentSkipListMap全面解析

Java并发基础:ConcurrentSkipListSet全面解析!

Java并发基础:SynchronousQueue全面解析!

相关文章:

Java并发基础:原子类之AtomicIntegerFieldUpdater全面解析

本文概要 AtomicIntegerFieldUpdater类提供了一种高效、简洁的方式来原子性地更新对象的volatile字段&#xff0c;无需使用重量级的锁机制&#xff0c;它通过基于反射的API实现了细粒度的并发控制&#xff0c;提升了多线程环境下的性能表现。 AtomicIntegerFieldUpdater核心概…...

普中51单片机学习(串口通信)

串口通信 原理 计算机通信是将计算机技术和通信技术的相结合&#xff0c;完成计算机与外部设备或计算机与计算机之间的信息交换 。可以分为两大类&#xff1a;并行通信与串行通信。并行通信通常是将数据字节的各位用多条数据线同时进行传送 。控制简单、传输速度快&#xff1…...

【ArcGIS】利用高程进行坡度分析

在ArcGIS中利用高程进行坡度分析 坡度ArcGIS实操参考 坡度 坡度是地表单元陡缓的程度&#xff0c;通常把坡面的垂直高度和水平距离的比值称为坡度。 坡度的表示方法有百分比法、度数法、密位法和分数法四种&#xff0c;其中以百分比法和度数法较为常用。 &#xff08;1&#…...

递归读取文件夹下的所有文件

水一篇文章 &#x1f436; 代码 package file;import org.apache.commons.lang3.StringUtils; import org.junit.Test;import java.io.File; import java.util.Objects;/*** FlattenDirFiles** author allens* date 2024/2/19*/ public class FlattenDirFiles {// 文件数量pri…...

phpspreadsheet导出数据和图片到excel

仅作记录&#xff0c;废话不多说 前提是已经安装了phpspreadsheet &#xff08; composer require phpoffice/phpspreadsheet &#xff09; 一、 数据拼装&#xff0c;调用excel类 <?php /*** 电子台账* Date: 2023/4/20* Time: 17:28*/namespace app\store\controlle…...

Seata的 TCC 模式

目录 概述 使用 依赖与配置 代码 概述 TCC 模式是一种侵入式的分布式事务解决方案&#xff0c;它不依赖于数据库的事务&#xff0c;而是要求开发者自定义完成 预提交、提交、回滚的方法逻辑。因此&#xff0c;它是一个种偏 复杂、灵活、有侵入性 的分布式事务处理方案。 De…...

Vue全局指令防止重复点击(等待请求)

继《vue之全局请求loading》之后&#xff0c;总觉得全局loading有时候不太…友好&#xff0c;所以总想将loading加到被点击的元素上面&#xff0c;于是乎就想到了点击事件与请求方法相关联&#xff0c;本想重写组件的click方法&#xff0c;但是这样对组件的影响太大&#xff0c…...

数据库索引面试的相关问题

查看索引的执行计划 索引失效的情况 1、索引列上做了计算&#xff0c;函数&#xff0c;类型转换等操作。索引失效是因为查询过程需要扫描整个索引并回表。代价高于直接全表扫描。 Like匹配使用了前缀匹配符“%abc” 字符串不加引号导致类型转换。 原因&#xff1a; 常见索…...

Spring启动生命周期

Spring Boot 生命周期详解 Spring Boot 应用程序的生命周期包含几个阶段&#xff0c;每个阶段都有特定的事件和钩子&#xff0c;允许开发者在应用程序的不同生命周期阶段插入自定义逻辑。以下是 Spring Boot 生命周期的主要阶段和对应的事件&#xff1a; 准备阶段&#xff1a;…...

瑞芯微RK3568芯片介绍

RK3568简介 RK3568是一款由瑞芯微电子有限公司&#xff08;Rockchip Electronics Co. Ltd&#xff09;推出的高性能多媒体处理器&#xff0c;定位于中高端市场&#xff0c;被广泛应用于平板电脑、多媒体盒子、电子看板和工业控制等领域。以下是对这款芯片的简要介绍&#xff1…...

15.一种坍缩式的简单——组合模式详解

当曾经的孩子们慢慢步入社会才知道&#xff0c;那年味渐淡的春节就像是疾驰在人生路上的暂停键。 它允许你在隆隆的鞭炮声中静下心来&#xff0c;瞻前顾后&#xff0c;怅然若失。 也允许你在寂静的街道上屏气凝神&#xff0c;倾听自己胸腔里的那团人声鼎沸。 孩子们会明白的&am…...

Node.js的debug模块源码分析及在harmonyOS平台移植

Debug库 是一个小巧但功能强大的 JavaScript 调试工具库&#xff0c;可以帮助开发人员更轻松地进行调试&#xff0c;以便更快地发现和修复问题。它的主要特点是可以轻松地添加调试日志语句&#xff0c;同时在不需要调试时可以轻松地禁用它们&#xff0c;以避免在生产环境中对性…...

【Crypto | CTF】BUUCTF RSA2

天命&#xff1a;密码学越来越难了&#xff0c;看别人笔记都不知道写啥 天命&#xff1a;莫慌&#xff0c;虽然我不会推演法&#xff0c;但我可以用归纳法 虽然我不知道解题的推演&#xff0c;但我可以背公式啊哈哈哈 虽然我不会这题&#xff0c;但是我也能做出来 公式我不知…...

单片机学习笔记---红外遥控红外遥控电机调速(完结篇)

目录 低电平触发中断和下降沿触发中断的区别 红外遥控 Int0.c Int.h Timer0.c Timer0.h IR.c IR.h main.c 红外遥控电机调速 Timer1.c Timer.h Motor.c Motor.h main.c 上一节讲了红外发送和接收的工作原理&#xff0c;这一节开始代码演示&#xff01; 提前说…...

Linux第62步_备份移植好的所有的文件和文件夹

1、备份“my-tfa”目录下所有的文件和文件夹 1)、打开终端 输入“ls回车”&#xff0c;列出当前目录下所有的文件和文件夹 输入“cd linux回车”&#xff0c;切换“linux”目录下 输入“ls回车”&#xff0c;列出当前目录下所有的文件和文件夹 输入“cd atk-mp1/回车”&am…...

【xss跨站漏洞】xss漏洞前置知识点整理

xss漏洞成因 xss漏洞是一种前端javascript产生的漏洞。 我们网站基本都是会用到javascript编写一些东西&#xff0c;浏览器也能直接识别javascript。 如果有一个地方能够输入文字&#xff0c;但是他又没有过滤你的输入&#xff0c;那么自己或者他人看到你输入的javascript代…...

mac下mysql 常用命令

mysql启动命令 在Mac OS X启动和停止MySQL服务的命令&#xff0c; 启动MySQL服务 sudo /usr/local/mysql/support-files/mysql.server start 停止MySQL服务 sudo /usr/local/mysql/support-files/mysql.server stop 重启MySQL服务 sudo /usr/local/mysql/support-files/mys…...

2.21号qt

1.QMainWindow中常用的类 继承于QMainWindow类&#xff0c;原因该类提供了QWidget没有提供的成员函数。 菜单栏、工具栏、状态栏、浮动窗口&#xff08;铆接部件&#xff09;、核心部件 1.1 菜单栏 QMenuBar //创建菜单栏 QMenuBar 最多只能有一个 QMenuBar *mbar menu…...

什么是MVVM?MVC、MVP与MVVM模式的区别?

MVVM&#xff08;Model-View-ViewModel&#xff09;是一种软件架构模式&#xff0c;用于将用户界面&#xff08;View&#xff09;与业务逻辑&#xff08;Model&#xff09;分离&#xff0c;并通过ViewModel来连接两者。MVVM的目标是实现可测试性、可维护性和可复用性。 MVC&am…...

ElementUI组件的安装和使用

Element UI 是一款基于 Vue 2.0 的桌面端组件库&#xff0c;主要用于快速构建网站的前端部分。它提供了丰富的组件&#xff0c;如按钮、输入框、表格、标签页等&#xff0c;以及一些布局元素&#xff0c;如布局容器、分割线等。Element UI 的设计风格简洁&#xff0c;易于上手&…...

【Oracle APEX开发小技巧12】

有如下需求&#xff1a; 有一个问题反馈页面&#xff0c;要实现在apex页面展示能直观看到反馈时间超过7天未处理的数据&#xff0c;方便管理员及时处理反馈。 我的方法&#xff1a;直接将逻辑写在SQL中&#xff0c;这样可以直接在页面展示 完整代码&#xff1a; SELECTSF.FE…...

Zustand 状态管理库:极简而强大的解决方案

Zustand 是一个轻量级、快速和可扩展的状态管理库&#xff0c;特别适合 React 应用。它以简洁的 API 和高效的性能解决了 Redux 等状态管理方案中的繁琐问题。 核心优势对比 基本使用指南 1. 创建 Store // store.js import create from zustandconst useStore create((set)…...

ssc377d修改flash分区大小

1、flash的分区默认分配16M、 / # df -h Filesystem Size Used Available Use% Mounted on /dev/root 1.9M 1.9M 0 100% / /dev/mtdblock4 3.0M...

Mybatis逆向工程,动态创建实体类、条件扩展类、Mapper接口、Mapper.xml映射文件

今天呢&#xff0c;博主的学习进度也是步入了Java Mybatis 框架&#xff0c;目前正在逐步杨帆旗航。 那么接下来就给大家出一期有关 Mybatis 逆向工程的教学&#xff0c;希望能对大家有所帮助&#xff0c;也特别欢迎大家指点不足之处&#xff0c;小生很乐意接受正确的建议&…...

Java入门学习详细版(一)

大家好&#xff0c;Java 学习是一个系统学习的过程&#xff0c;核心原则就是“理论 实践 坚持”&#xff0c;并且需循序渐进&#xff0c;不可过于着急&#xff0c;本篇文章推出的这份详细入门学习资料将带大家从零基础开始&#xff0c;逐步掌握 Java 的核心概念和编程技能。 …...

React---day11

14.4 react-redux第三方库 提供connect、thunk之类的函数 以获取一个banner数据为例子 store&#xff1a; 我们在使用异步的时候理应是要使用中间件的&#xff0c;但是configureStore 已经自动集成了 redux-thunk&#xff0c;注意action里面要返回函数 import { configureS…...

代码随想录刷题day30

1、零钱兑换II 给你一个整数数组 coins 表示不同面额的硬币&#xff0c;另给一个整数 amount 表示总金额。 请你计算并返回可以凑成总金额的硬币组合数。如果任何硬币组合都无法凑出总金额&#xff0c;返回 0 。 假设每一种面额的硬币有无限个。 题目数据保证结果符合 32 位带…...

jmeter聚合报告中参数详解

sample、average、min、max、90%line、95%line,99%line、Error错误率、吞吐量Thoughput、KB/sec每秒传输的数据量 sample&#xff08;样本数&#xff09; 表示测试中发送的请求数量&#xff0c;即测试执行了多少次请求。 单位&#xff0c;以个或者次数表示。 示例&#xff1a;…...

Vite中定义@软链接

在webpack中可以直接通过符号表示src路径&#xff0c;但是vite中默认不可以。 如何实现&#xff1a; vite中提供了resolve.alias&#xff1a;通过别名在指向一个具体的路径 在vite.config.js中 import { join } from pathexport default defineConfig({plugins: [vue()],//…...

PHP 8.5 即将发布:管道操作符、强力调试

前不久&#xff0c;PHP宣布了即将在 2025 年 11 月 20 日 正式发布的 PHP 8.5&#xff01;作为 PHP 语言的又一次重要迭代&#xff0c;PHP 8.5 承诺带来一系列旨在提升代码可读性、健壮性以及开发者效率的改进。而更令人兴奋的是&#xff0c;借助强大的本地开发环境 ServBay&am…...