【高薪程序员必看】万字长文拆解Java并发编程!(6-1):从CAS无锁机制到Atomic原子类实战指南

简介: 🌟 ​🌟今天给大家带来的是 ​💻⚡在这篇文章中,我们将一起探索:🔹 ​的底层原理,它是如何通过 ​实现无锁并发的?🔹 ​的终极对决,为什么高并发场景下CAS性能更优?🔹 ​的陷阱与解决方案——和实战演示!🔹 ​​(LongAdder等)的使用场景与性能对比🔹 危险的 ​黑魔法:为什么阿里禁止使用却又是并发库的基石?无论你是:✅ ​​(BATJ高频考点)✅ ​​(如何设计百万级计数器)✅ ​​(从Java代码到CPU指令的全链路分析)这篇文章都会让你收获满满!✨。

  image.gif 编辑

🌟 大家好,我是摘星! 🌟

今天给大家带来的是 《Java高并发编程核心:CAS无锁机制与原子类深度解析》 💻⚡

在这篇文章中,我们将一起探索:

🔹 CAS(Compare-And-Swap) 的底层原理,它是如何通过 CPU指令 实现无锁并发的?

🔹 乐观锁 vs 悲观锁 的终极对决,为什么高并发场景下CAS性能更优?

🔹 ABA问题 的陷阱与解决方案——AtomicStampedReferenceAtomicMarkableReference实战演示!

🔹 Atomic原子类全家桶AtomicIntegerLongAdder等)的使用场景与性能对比

🔹 危险的 Unsafe 黑魔法:为什么阿里禁止使用却又是并发库的基石?

无论你是:

面试突击(BATJ高频考点)

性能调优(如何设计百万级计数器)

底层原理控(从Java代码到CPU指令的全链路分析)

这篇文章都会让你收获满满!✨

你在项目中用过CAS吗?遇到过哪些坑?欢迎评论区分享~ 👇

目录

6. 无锁并发-乐观锁

6.1. CAS

6.2. ABA问题解决

6.3. 原子类


6. 无锁并发-乐观锁

6.1. CAS

CAS(Compare and Swap)是一种无锁的并发控制机制,常用于多线程环境下的原子操作。

CAS操作包含三个参数:变量当前值、变量期望值、更新值,具体步骤如下:

  1. 读取变量当前值
  2. 比较变量当前值和期望值是否相等
  3. 如果相等,说明变量的值没有被其他线程修改过,将更新值赋给变量
  4. 如果不相等,说明变量的值已经被其他线程修改,CAS操作失败

要素

说明

操作原理

比较当前值(V)、期望值(E)、新值(N),若 V==E 则更新为 N,否则失败

底层依赖

volatile(保证可见性) + CPU原子指令(如x86的CMPXCHG

乐观锁思想

假设无竞争,失败时重试而非阻塞

CAS操作依赖于volatile的可见性来读取变量当前值,并且依赖volatile的禁止重排序的特性来保证原子性

CAS是基于乐观锁的思想,假设别的线程不会修改共享资源,就算修改了也没关系,乐观锁在修改共享资源时会对资源进行一次检查,如果没有被修改过则直接更新,如果被修改过就重试。因此效率高,因为它避免了加锁和解锁的操作,竞争失败时也不会发生线程上下文切换的情况。但是它也存在一些问题,比如ABA问题,以及激烈的竞争情况下会导致不断重试。

在Java中,CAS是使用java.util.concurrent.atomic包中的原子类实现的,常用的CAS操作是由AtomicXXX类提供的,比如AtomicIntegerAtomicLong等。这些类提供了一系列的原子操作方法,如compareAndSet()getAndIncrement()等,来实现无锁的并发控制。

对比维度

CAS(乐观锁)

悲观锁(如synchronized)

线程阻塞

❌ 无阻塞,失败时自旋重试

✅ 竞争失败时线程挂起

适用场景

低冲突、短耗时操作(如计数器)

高冲突、长耗时操作(如数据库事务)

ABA问题

存在(需版本号解决)

不存在

性能开销

⚡ 轻量级(无上下文切换)

⚠️ 重量级(锁竞争、唤醒开销)

6.2. ABA问题解决

CAS 存在一个经典的问题,就是 ABA 问题,即在多线程环境下,如果一个值原来是 A,后来变成了 B,然后又变回 A,但是CAS操作仍然操作成功,这是不被允许的。

为了解决 CAS 的 ABA 问题,Java 并发包提供了以下两种方法:

  1. 使用 AtomicStampedReferenceAtomicStampedReference 是一个带有标记的引用类,它对应的共享变量是一个包装了 value 和 stamp(标记)的对象。通过比较和交换引用值和标记值,AtomicStampedReference 可以解决 ABA 问题。每当共享变量发生变化时,都需要更新标记值,这样即使值发生了 ABA 的变化,标记值也会发生变化,从而保证 CAS 可以正确地判断出是否发生了变化。
  2. 使用 AtomicMarkableReferenceAtomicMarkableReference 是另一个带有标记的引用类,相比于 AtomicStampedReference,它使用了一个 boolean 类型的标记来解决 ABA 问题。与 AtomicStampedReference 类似,每当共享变量发生变化时,都需要更新标记值。通过比较和交换引用值和标记值,AtomicMarkableReference 可以避免 ABA 问题。

方案

原理

代码示例

AtomicStampedReference

通过int stamp版本号标记状态变化

java<br>AtomicStampedReference<Integer> ref = new AtomicStampedReference<>(100, 0);<br>ref.compareAndSet(100, 200, 0, 1);<br>

AtomicMarkableReference

通过boolean mark标记状态(简化版)

java<br>AtomicMarkableReference<Integer> ref = new AtomicMarkableReference<>(100, false);<br>ref.compareAndSet(100, 200, false, true);<br>

6.3. 原子类

在Java并发编程中,原子类是一组提供原子操作的类。原子操作是指不可分割的操作,不会被其他线程中断,也不会被中断其他操作。Java提供了一些原子类,用于处理并发编程中的线程安全问题。

以下是一些常见的原子类:

  1. AtomicBoolean:提供了用于原子操作布尔值的方法,比如get()、set()、getAndSet()和compareAndSet()等。
  2. AtomicInteger和AtomicLong:分别提供了原子操作整型和长整型的方法。例如,incrementAndGet()、decrementAndGet()、getAndIncrement()、getAndDecrement()、getAndAdd()等。
  3. AtomicReference:用于原子操作引用类型的变量。它提供了get()、set()、getAndSet()和compareAndSet()等方法。
  4. AtomicReferenceArray:类似于AtomicReference,但是用于数组。它提供了对数组元素进行原子操作的方法,如get()、set()、getAndSet()和compareAndSet()等。
  5. AtomicIntegerArray和AtomicLongArray:类似于AtomicInteger和AtomicLong,但是用于整型数组和长整型数组。提供了对数组元素进行原子操作的方法,如get()、set()、getAndSet()和compareAndSet()等。

类别

典型类

适用场景

基本类型

AtomicInteger/Long

计数器、序号生成(如i++原子化)

引用类型

AtomicReference

对象引用的原子更新(如单例模式)

数组类型

AtomicIntegerArray

并发安全的数组操作

字段更新器

AtomicIntegerFieldUpdater

已存在类的volatile字段原子更新(减少对象开销)

AtomicInteger API

方法

作用

等效代码(非原子)

i.get()

获取当前值

int val = i;

i.incrementAndGet()

++i(先增后取)

return ++i;

i.getAndIncrement()

i++(先取后增)

return i++;

i.updateAndGet(x -> x*2)

原子运算(如乘2)

i = i * 2; return i;

🌟 感谢大家看到这里!我是摘星,我们下期再见! 🌟


🔹 如果这篇文章对你有帮助,欢迎点赞❤️ + 收藏⭐,让更多小伙伴看到~

🔹 有任何问题或想法,欢迎在评论区留言,我会一一回复!

🔹 关注我,解锁更多 Java 高并发 | 分布式 | JVM 调优 的深度解析!

相关文章
|
5月前
|
安全 Java 开发者
告别NullPointerException:Java Optional实战指南
告别NullPointerException:Java Optional实战指南
323 119
|
7月前
|
缓存 前端开发 Java
基于最新 Java 技术栈的在线任务管理系统开发实战详解
本项目基于最新Java技术栈开发在线任务管理系统,涵盖任务创建、分配、跟踪、统计等功能。采用Spring Boot 3.2.x、React 18、PostgreSQL 16等主流技术,详解项目架构设计、核心功能实现及部署流程,助力掌握现代Java全栈开发技能。
439 6
|
7月前
|
Java 关系型数据库 数据库
Java 项目实战教程从基础到进阶实战案例分析详解
本文介绍了多个Java项目实战案例,涵盖企业级管理系统、电商平台、在线书店及新手小项目,结合Spring Boot、Spring Cloud、MyBatis等主流技术,通过实际应用场景帮助开发者掌握Java项目开发的核心技能,适合从基础到进阶的学习与实践。
1100 3
|
7月前
|
Java API Maven
2025 Java 零基础到实战最新技术实操全攻略与学习指南
本教程涵盖Java从零基础到实战的全流程,基于2025年最新技术栈,包括JDK 21、IntelliJ IDEA 2025.1、Spring Boot 3.x、Maven 4及Docker容器化部署,帮助开发者快速掌握现代Java开发技能。
1492 1
|
6月前
|
存储 前端开发 Java
【JAVA】Java 项目实战之 Java Web 在线商城项目开发实战指南
本文介绍基于Java Web的在线商城技术方案与实现,涵盖三层架构设计、MySQL数据库建模及核心功能开发。通过Spring MVC + MyBatis + Thymeleaf实现商品展示、购物车等模块,提供完整代码示例,助力掌握Java Web项目实战技能。(238字)
764 0
|
7月前
|
消息中间件 Java Kafka
Java 事件驱动架构设计实战与 Kafka 生态系统组件实操全流程指南
本指南详解Java事件驱动架构与Kafka生态实操,涵盖环境搭建、事件模型定义、生产者与消费者实现、事件测试及高级特性,助你快速构建高可扩展分布式系统。
374 7
|
6月前
|
Java 开发者
Java并发编程:CountDownLatch实战解析
Java并发编程:CountDownLatch实战解析
548 100
|
7月前
|
数据采集 JSON Java
Java爬虫获取1688店铺所有商品接口数据实战指南
本文介绍如何使用Java爬虫技术高效获取1688店铺商品信息,涵盖环境搭建、API调用、签名生成及数据抓取全流程,并附完整代码示例,助力市场分析与选品决策。
|
7月前
|
消息中间件 Java 数据库
Java 基于 DDD 分层架构实战从基础到精通最新实操全流程指南
本文详解基于Java的领域驱动设计(DDD)分层架构实战,结合Spring Boot 3.x、Spring Data JPA 3.x等最新技术栈,通过电商订单系统案例展示如何构建清晰、可维护的微服务架构。内容涵盖项目结构设计、各层实现细节及关键技术点,助力开发者掌握DDD在复杂业务系统中的应用。
1421 0
|
8月前
|
监控 Java API
现代 Java IO 高性能实践从原理到落地的高效实现路径与实战指南
本文深入解析现代Java高性能IO实践,涵盖异步非阻塞IO、操作系统优化、大文件处理、响应式网络编程与数据库访问,结合Netty、Reactor等技术落地高并发应用,助力构建高效可扩展的IO系统。
249 0