一:原子类
Java从JDK 1.5开始提供了java.util.concurrent.atomic包,这个包中的原子操作类提供了一种用法简单、性能高效、线程安全地更新一个变量的方式。在atomic包里一共提供了17个类,属于4种类型的原子更新方式,分别是原子更新基本类型、原子更新引用类型、原子更新属性、原子更新数组。
atomic包里的类基本都是使用Unsafe实现的包装类,而Unsafe类提供了如下3个CAS方法。这三个方法都带有4个参数,这些参数依次指代:对象、成员变量、期望的值、更新的值。
- compareAndSwapInt(Object var1, long var2, int var4, int var5)
- compareAndSwapLong(Object var1, long var2, long var4, long var6)
- compareAndSwapObject(Object var1, long var2, Object var4, Object var5)
原子更新基本类型
包括3个类:AtomicInteger、AtomicLong、AtomicBoolean,这3个类提供的方法几乎一模一样。
其他基本类型的变量,如char、float、double,可以先转换为整型,然后再进行原子操作。例如,AtomicBoolean就是先把Boolean转换成整型,再使用compareAndSwapInt进行CAS操作。
原子更新引用类型
包括3个类:
- AtomicReference:原子更新引用类型;
- AtomicStampedReference:原子更新带有标记位的引用类型;
- AtomicMarkableReference:原子更新带有标记位的引用类型。
其中,后两个类用于解决ABA问题,区别是标记位的类型不同,它们分别以整数/布尔类型做标记位。
原子更新属性
包括3个类:AtomicIntegerFieldUpdater、AtomicLongFieldUpdater、AtomicReferenceFieldUpdater。
如果一个类是自己编写的,则可以在编写的时候把成员变量定义为Atomic类型。但如果是一个已经有的
类,在不能更改其源代码的情况下,要想实现对其成员变量的原子操作,就需要使用上述三个类。
原子更新数组
包括3个类:AtomicIntegerArray、AtomicLongArray、AtomicReferenceArray,可以通过原子的方式
更新数组里的某个元素。
Striped64
从JDK 8开始,Java提供了Striped64类,该类共包含4个子类,分别是LongAdder、LongAccumulator、
DoubleAdder、DoubleAccumulator。之所以提供这些类,是为了解决在高并发场景下多个线程对同一个
变量进行CAS操作的性能问题。