Java并发编程实战 Day 10:原子操作类详解

【Java并发编程实战 Day 10】原子操作类详解

开篇

这是“Java并发编程实战”系列的第10天,我们将深入探讨原子操作类的核心技术——CAS原理、ABA问题以及原子类的实现机制。通过理论结合代码实践的方式,帮助读者理解并掌握如何在实际工作中高效使用原子操作类。


理论基础

原子操作是并发编程中的重要概念,其核心在于无锁化操作。以下是关键点解析:

1. CAS(Compare-And-Swap)原理

CAS是一种基于硬件支持的原子指令,用于在多线程环境下实现无锁操作。其工作流程如下:

  • 比较:检查内存中的值是否等于预期值。
  • 交换:如果相等,则将内存中的值替换为目标值。
  • 失败重试:如果不相等,则重新尝试操作。

CAS避免了传统锁的开销,但可能因频繁自旋导致CPU资源浪费。

2. ABA问题

ABA问题是CAS操作中常见的陷阱。例如:

  • 线程A读取某变量值为A。
  • 线程B将该变量从A改为B,再改回A。
  • 线程A再次执行CAS时,误以为变量未被修改。

解决方法是引入版本号或时间戳,如AtomicStampedReference

3. 原子类实现

JDK提供了丰富的原子类,包括AtomicIntegerAtomicLongAtomicReference等。它们底层依赖Unsafe类提供的CAS操作。


适用场景

原子操作类适用于以下场景:

  • 计数器:如统计在线用户数量。
  • 状态标志位:如控制任务执行状态。
  • 共享数据更新:如缓存中的数据更新。

这些场景通常涉及高频次的读写操作,且对性能要求较高。


代码实践

以下是一个完整的代码示例,演示如何使用AtomicInteger实现计数器功能,并对比普通锁的性能。

import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;public class AtomicExample {private static final int THREAD_COUNT = 1000;private static final int ITERATIONS = 10000;// 使用AtomicInteger实现计数器private static AtomicInteger atomicCounter = new AtomicInteger(0);// 使用ReentrantLock实现计数器private static int lockCounter = 0;private static Lock lock = new ReentrantLock();public static void main(String[] args) throws InterruptedException {// 测试AtomicInteger性能Thread[] atomicThreads = new Thread[THREAD_COUNT];for (int i = 0; i < THREAD_COUNT; i++) {atomicThreads[i] = new Thread(() -> {for (int j = 0; j < ITERATIONS; j++) {atomicCounter.incrementAndGet();}});}long start = System.currentTimeMillis();for (Thread t : atomicThreads) t.start();for (Thread t : atomicThreads) t.join();long atomicTime = System.currentTimeMillis() - start;// 测试ReentrantLock性能Thread[] lockThreads = new Thread[THREAD_COUNT];for (int i = 0; i < THREAD_COUNT; i++) {lockThreads[i] = new Thread(() -> {for (int j = 0; j < ITERATIONS; j++) {lock.lock();try {lockCounter++;} finally {lock.unlock();}}});}start = System.currentTimeMillis();for (Thread t : lockThreads) t.start();for (Thread t : lockThreads) t.join();long lockTime = System.currentTimeMillis() - start;// 输出结果System.out.println("Atomic Counter Value: " + atomicCounter.get());System.out.println("Lock Counter Value: " + lockCounter);System.out.println("Atomic Time: " + atomicTime + " ms");System.out.println("Lock Time: " + lockTime + " ms");}
}

运行上述代码,可以观察到AtomicInteger的性能显著优于ReentrantLock


实现原理

原子类的底层实现依赖于Unsafe类提供的CAS操作。以AtomicInteger为例:

public final int incrementAndGet() {return unsafe.getAndAddInt(this, valueOffset, 1) + 1;
}

valueOffset是变量在内存中的偏移量,unsafe.getAndAddInt通过CAS指令完成原子性操作。


性能测试

以下是性能测试结果对比:

方法平均耗时(ms)
AtomicInteger150
ReentrantLock400

由此可见,原子操作类在高并发场景下的性能优势明显。


最佳实践

  1. 优先使用原子类:在无需复杂同步逻辑的情况下,优先选择原子类。
  2. 注意ABA问题:在涉及状态变更的场景中,使用AtomicStampedReference
  3. 合理设置自旋次数:避免因频繁自旋导致CPU资源浪费。

案例分析

假设我们需要实现一个秒杀系统中的库存扣减功能。传统方式可能使用synchronizedReentrantLock,但性能较差。通过引入AtomicInteger,可以显著提升吞吐量。

private static AtomicInteger stock = new AtomicInteger(100);public static boolean deductStock() {while (true) {int current = stock.get();if (current <= 0) return false; // 库存不足if (stock.compareAndSet(current, current - 1)) return true; // 扣减成功}
}

总结

本篇文章详细讲解了CAS原理、ABA问题及原子类的实现机制,并通过代码实践展示了其在高并发场景下的应用价值。核心知识点包括:

  • CAS操作的工作原理
  • ABA问题及解决方案
  • 原子类的底层实现

下一篇文章将介绍并发设计模式,敬请期待!

参考资料

  1. Java官方文档
  2. 《Java并发编程实战》
  3. 《深入理解Java虚拟机》

核心技能总结

通过本篇文章,您学会了如何使用原子操作类解决高并发场景下的线程安全问题,掌握了CAS原理及其实现机制,并了解了最佳实践。这些技能可直接应用于实际项目中,如库存管理、计数器实现等场景。

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

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

相关文章

瀚文机械键盘固件开发详解:HWKeyboard.h文件解析与应用

【手把手教程】从零开始的机械键盘固件开发&#xff1a;HWKeyboard.h详解 前言 大家好&#xff0c;我是键盘DIY爱好者Despacito0o&#xff01;今天想和大家分享我开发的机械键盘固件核心头文件HWKeyboard.h的设计思路和技术要点。这个项目是我多年来对键盘固件研究的心血结晶…

2048游戏的技术实现分析-完全Java和Processing版

目录 简介Processing库基础项目构建指南项目结构核心数据结构游戏核心机制图形界面实现性能优化代码详解设计模式分析测试策略总结与展望简介 2048是一款由Gabriele Cirulli开发的经典益智游戏。本文将深入分析其Java实现版本的技术细节。该实现使用了Processing库来创建图形界…

Spring Boot + Elasticsearch + HBase 构建海量数据搜索系统

Spring Boot Elasticsearch HBase 构建海量数据搜索系统 &#x1f4d6; 目录 1. 系统需求分析2. 系统架构设计3. Elasticsearch 与 HBase 集成方案4. Spring Boot 项目实现5. 大规模搜索系统最佳实践 项目概述 本文档提供了基于 Spring Boot、Elasticsearch 和 HBase 构建…

【iOS】YYModel源码解析

YYModel源码解析 文章目录 YYModel源码解析前言YYModel性能优势YYModel简介YYClassInfo解析YYClassIvarInfo && objc_ivarYYClassMethodInfo && objc_methodYYClassPropertyInfo && property_tYYClassInfo && objc_class YYClassInfo的初始化细…

宇树科技更名“股份有限公司”深度解析:机器人企业IPO前奏与资本化路径

从技术落地到资本跃迁&#xff0c;拆解股改背后的上市逻辑与行业启示 核心事件&#xff1a;股改释放的上市信号 2025年5月28日&#xff0c;杭州宇树科技有限公司正式更名“杭州宇树科技股份有限公司”&#xff0c;市场主体类型变更为“股份有限公司”。尽管官方称为常规运营调…

Android Native 内存泄漏检测全解析:从原理到工具的深度实践

引言 Android应用的内存泄漏不仅发生在Java/Kotlin层&#xff0c;Native&#xff08;C/C&#xff09;层的泄漏同样普遍且隐蔽。由于Native内存不受Java虚拟机&#xff08;JVM&#xff09;管理&#xff0c;泄漏的内存无法通过GC自动回收&#xff0c;长期积累会导致应用内存占用…

Vortex GPGPU的github流程跑通与功能模块波形探索(四)

文章目录 前言一、demo的输入文件二、trace_csv三、2个值得注意的点3.1 csv指令表格里面的tmask&#xff1f;3.2 rtlsim和simx的log文件&#xff1f; 总结 前言 跟着前面那篇最后留下的几个问题接着把输出波形文件和csv文件的输入、输出搞明白&#xff01; 一、demo的输入文件…

UnityPSD文件转UI插件Psd2UnityuGUIPro3.4.0u2017.4.2介绍:Unity UI设计的高效助手

UnityPSD文件转UI插件Psd2UnityuGUIPro3.4.0u2017.4.2介绍&#xff1a;Unity UI设计的高效助手 【下载地址】UnityPSD文件转UI插件Psd2UnityuGUIPro3.4.0u2017.4.2介绍 这款开源插件将PSD文件无缝转换为Unity的UI元素&#xff0c;极大提升开发效率。它支持一键转换&#xff0c;…

力扣100题之128. 最长连续序列

方法1 使用了hash 方法思路 使用哈希集合&#xff1a;首先将数组中的所有数字存入一个哈希集合中&#xff0c;这样可以在 O(1) 时间内检查某个数字是否存在。 寻找连续序列&#xff1a;遍历数组中的每一个数字&#xff0c;对于每一个数字&#xff0c; 检查它是否是某个连续序列…

Java爬虫技术详解:原理、实现与优势

一、什么是网络爬虫&#xff1f; 网络爬虫&#xff08;Web Crawler&#xff09;&#xff0c;又称网络蜘蛛或网络机器人&#xff0c;是一种自动化程序&#xff0c;能够按照一定的规则自动浏览和抓取互联网上的信息。爬虫技术是大数据时代获取网络数据的重要手段&#xff0c;广泛…

神经网络与深度学习 网络优化与正则化

1.网络优化存在的难点 &#xff08;1&#xff09;结构差异大&#xff1a;没有通用的优化算法&#xff1b;超参数多 &#xff08;2&#xff09;非凸优化问题&#xff1a;参数初始化&#xff0c;逃离局部最优 &#xff08;3&#xff09;梯度消失&#xff08;爆炸&#xff09; …

【汇编逆向系列】二、函数调用包含单个参数之整型-ECX寄存器,LEA指令

目录 一. 汇编源码 二. 汇编分析 1. ECX寄存器 2. 栈位置计算​ 3. 特殊指令深度解析 三、 汇编转化 一. 汇编源码 single_int_param:0000000000000040: 89 4C 24 08 mov dword ptr [rsp8],ecx0000000000000044: 57 push rdi0000…

Linux进程替换以及exec六大函数运用

文章目录 1.进程替换2.替换过程3.替换函数exec3.1命名解释 4.细说6个exe函数execl函数execvexeclp、execvpexecle、execve 1.进程替换 fork&#xff08;&#xff09;函数在创建子进程后&#xff0c;子进程如果想要执行一个新的程序&#xff0c;就可以使用进程的程序替换来完成…

Selenium操作指南(全)

&#x1f345; 点击文末小卡片&#xff0c;免费获取软件测试全套资料&#xff0c;资料在手&#xff0c;涨薪更快 大家好&#xff0c;今天带大家一起系统的学习下模拟浏览器运行库Selenium&#xff0c;它是一个用于Web自动化测试及爬虫应用的重要工具。 Selenium测试直接运行在…

结构性设计模式之Facade(外观)设计模式

结构性设计模式之Facade&#xff08;外观&#xff09;设计模式 前言&#xff1a; 外观模式&#xff1a;用自己的话理解就是用户看到是一个总体页面&#xff0c;比如xx报名系统页面。里面有历年真题模块、报名模块、教程模块、首页模块… 做了一个各个模块的合并&#xff0c;对…

RabbitMQ实用技巧

RabbitMQ是一个流行的开源消息中间件&#xff0c;广泛用于实现消息传递、任务分发和负载均衡。通过合理使用RabbitMQ的功能&#xff0c;可以显著提升系统的性能、可靠性和可维护性。本文将介绍一些RabbitMQ的实用技巧&#xff0c;包括基础配置、高级功能及常见问题的解决方案。…

Linux(10)——第二个小程序(自制shell)

目录 ​编辑 一、引言与动机 &#x1f4dd;背景 &#x1f4dd;主要内容概括 二、全局数据 三、环境变量的初始化 ✅ 代码实现 四、构造动态提示符 ✅ 打印提示符函数 ✅ 提示符生成函数 ✅获取用户名函数 ✅获取主机名函数 ✅获取当前目录名函数 五、命令的读取与…

环境变量深度解析:从配置到内核的全链路指南

文章目录 一、基础概念与核心作用二、常见环境变量三、操作指南&#xff1a;从查看、修改到调试3.1 快速查询3.2 PATH 原理与配置实践3.2.1 命令执行机制3.2.2 路径管理策略 四、编程接口与内存模型4.1 环境变量的内存结构4.2 C 语言访问方式4.2.1 直接访问&#xff08;main 参…

结合Jenkins、Docker和Kubernetes等主流工具,部署Spring Boot自动化实战指南

基于最佳实践的Spring Boot自动化部署实战指南,结合Jenkins、Docker和Kubernetes等主流工具,提供从环境搭建到生产部署的完整流程: 一、环境准备与工具选型​​ ​​1.基础设施​​ ​​Jenkins服务器​​:安装Jenkins LTS版本,配置JDK(推荐JDK 11+)及Maven/Gradle插…

动态规划---股票问题

1.在推状态转移方程的途中&#xff0c;箭头的起始点表示前一天的状态&#xff0c;箭头的终点是当天的状态 2.当动态规划中涉及到多状态&#xff0c;且状态之间可以相互转换&#xff0c;要画图去分析 1.买卖股票的最佳时机含冷冻期 题目链接&#xff1a;309. 买卖股票的最佳时机…