Atomic 原子类介绍
Atomic
翻译成中文是“原子”的意思。在化学上,原子是构成物质的最小单位,在化学反应中不可分割。在编程中,Atomic
指的是一个操作具有原子性,即该操作不可分割、不可中断。即使在多个线程同时执行时,该操作要么全部执行完成,要么不执行,不会被其他线程看到部分完成的状态。
原子类简单来说就是具有原子性操作特征的类。
java.util.concurrent.atomic
包中的 Atomic
原子类提供了一种线程安全的方式来操作单个变量。
Atomic
类依赖于 CAS(Compare-And-Swap,比较并交换)乐观锁来保证其方法的原子性,而不需要使用传统的锁机制(如 synchronized
块或 ReentrantLock
)。
这篇文章我们只介绍 Atomic 原子类的概念,具体实现原理可以阅读笔者写的这篇文章:CAS 详解。
根据操作的数据类型,可以将 JUC 包中的原子类分为 4 类:
1、基本类型
使用原子的方式更新基本类型
AtomicInteger
:整型原子类AtomicLong
:长整型原子类AtomicBoolean
:布尔型原子类
2、数组类型
使用原子的方式更新数组里的某个元素
AtomicIntegerArray
:整型数组原子类AtomicLongArray
:长整型数组原子类AtomicReferenceArray
:引用类型数组原子类
3、引用类型
AtomicReference
:引用类型原子类AtomicMarkableReference
:原子更新带有标记的引用类型。该类将 boolean 标记与引用关联起来,也可以解决使用 CAS 进行原子更新时可能出现的 ABA 问题。AtomicStampedReference
:原子更新带有版本号的引用类型。该类将整数值与引用关联起来,可用于解决原子的更新数据和数据的版本号,可以解决使用 CAS 进行原子更新时可能出现的 ABA 问题。
🐛 修正(参见:issue#626) : AtomicMarkableReference
不能解决 ABA 问题。
4、对象的属性修改类型
AtomicIntegerFieldUpdater
:原子更新整型字段的更新器AtomicLongFieldUpdater
:原子更新长整型字段的更新器AtomicReferenceFieldUpdater
:原子更新引用类型里的字段
基本类型原子类
使用原子的方式更新基本类型
AtomicInteger
:整型原子类AtomicLong
:长整型原子类AtomicBoolean
:布尔型原子类
上面三个类提供的方法几乎相同,所以我们这里以 AtomicInteger
为例子来介绍。
AtomicInteger
类常用方法 :
1 | public final int get() //获取当前的值 |
AtomicInteger
类使用示例 :
1 | // 初始化 AtomicInteger 对象,初始值为 0 |
输出:
1 | tempValue: 0; atomicInt: 3 |
数组类型原子类
使用原子的方式更新数组里的某个元素
AtomicIntegerArray
:整形数组原子类AtomicLongArray
:长整形数组原子类AtomicReferenceArray
:引用类型数组原子类
上面三个类提供的方法几乎相同,所以我们这里以 AtomicIntegerArray
为例子来介绍。
AtomicIntegerArray
类常用方法:
1 | public final int get(int i) //获取 index=i 位置元素的值 |
AtomicIntegerArray
类使用示例 :
1 | int[] nums = {1, 2, 3, 4, 5, 6}; |
输出:
1 | Initial values in AtomicIntegerArray: |
引用类型原子类
基本类型原子类只能更新一个变量,如果需要原子更新多个变量,需要使用 引用类型原子类。
AtomicReference
:引用类型原子类AtomicStampedReference
:原子更新带有版本号的引用类型。该类将整数值与引用关联起来,可用于解决原子的更新数据和数据的版本号,可以解决使用 CAS 进行原子更新时可能出现的 ABA 问题。AtomicMarkableReference
:原子更新带有标记的引用类型。该类将 boolean 标记与引用关联起来,也可以解决使用 CAS 进行原子更新时可能出现的 ABA 问题。
上面三个类提供的方法几乎相同,所以我们这里以 AtomicReference
为例子来介绍。
AtomicReference
类使用示例 :
1 | // Person 类 |
输出:
1 | Initial Person: Person{name='SnailClimb', age=22} |
AtomicStampedReference
类使用示例 :
1 | // 创建一个 AtomicStampedReference 对象,初始值为 "SnailClimb",初始版本号为 1 |
输出结果如下:
1 | Initial Reference: SnailClimb, Initial Stamp: 1 |
AtomicMarkableReference
类使用示例 :
1 | // 创建一个 AtomicMarkableReference 对象,初始值为 "SnailClimb",初始标记为 false |
输出结果如下:
1 | Initial Reference: SnailClimb, Initial Mark: false |
对象的属性修改类型原子类
如果需要原子更新某个类里的某个字段时,需要用到对象的属性修改类型原子类。
AtomicIntegerFieldUpdater
:原子更新整形字段的更新器AtomicLongFieldUpdater
:原子更新长整形字段的更新器AtomicReferenceFieldUpdater
:原子更新引用类型里的字段的更新器
要想原子地更新对象的属性需要两步。第一步,因为对象的属性修改类型原子类都是抽象类,所以每次使用都必须使用静态方法 newUpdater()创建一个更新器,并且需要设置想要更新的类和属性。第二步,更新的对象属性必须使用 public volatile 修饰符。
上面三个类提供的方法几乎相同,所以我们这里以 AtomicIntegerFieldUpdater
为例子来介绍。
AtomicIntegerFieldUpdater
类使用示例 :
1 | // Person 类 |
输出结果:
1 | Initial Person: Name: SnailClimb, Age: 22 |
参考
- 《Java 并发编程的艺术》