JUC AtomicIntegerArray源码解析 JDK8

news/2024/5/20 7:24:04 标签: java, JUC, 原子类, 并发编程

前言

AtomicIntegerArray提供了对数组元素的原子操作,与其他非数组的原子类相比,它的成员不是volatile的而是final的了。

JUC框架 系列文章目录

成员

java">    private static final Unsafe unsafe = Unsafe.getUnsafe();
    private static final int base = unsafe.arrayBaseOffset(int[].class);//获得第一个元素的起始位置
    private static final int shift;
    private final int[] array;  //注意不是volatile的

    static {
        int scale = unsafe.arrayIndexScale(int[].class);  //返回数组元素类型占几个字节,这里int是4字节
        if ((scale & (scale - 1)) != 0)  //scale必须为2的幂
            throw new Error("data type scale not a power of two");
        shift = 31 - Integer.numberOfLeadingZeros(scale);//返回二进制中第一个1之前的0的个数
    }

    private long checkedByteOffset(int i) {
        if (i < 0 || i >= array.length)
            throw new IndexOutOfBoundsException("index " + i);

        return byteOffset(i);
    }

    private static long byteOffset(int i) {
        return ((long) i << shift) + base;
    }
  • 既然是数组,所以要操作数组元素。而数组的内部存储方式有可能是先存size,再存各个元素,所以第一个元素的起始地址需要通过arrayBaseOffset确认。根据base地址,再加上 (元素个数-1) * 类型大小,就得到某个元素的起始地址。
  • arrayIndexScale得到数组元素类型的大小,这里是int是4字节,所以返回4。scale为4。
  • (scale & (scale - 1)) != 0这个判断,只有当scale为2的幂时,scale & (scale - 1)就会刚好等于0,然后判断为false。
  • scale和shift的关系是 s c a l e = 2 s h i f t scale = 2^{shift} scale=2shift。运算过程是这样的:因为scale是int型,所以使用Integer.numberOfLeadingZeros()(见注释),scale为4即0b100,int型总共32个bit,所以第一个1前面有29个0,numberOfLeadingZeros返回29。又因为使用的Integer.numberOfLeadingZeros(),所以前面那个数为32 -1。最终,31 - 29 = 2,即最终shift = 2。
  • 前面说到需要计算(元素个数-1) * 类型大小,现在看byteOffset函数的实现,你会发现 i < < s h i f t = i ∗ 2 s h i f t = i ∗ s c a l e i << shift = i*2^{shift } = i*scale i<<shift=i2shift=iscale,当然前提是左移没有超过最高位。而i传入的都是索引(元素的第几个减1),所以((long) i << shift) + base刚好就是(元素个数-1) * 类型大小了。
  • byteOffset函数,传入元素索引,返回这个数组元素的起始地址。

构造器

java">    public AtomicIntegerArray(int length) {
        array = new int[length];
    }

    public AtomicIntegerArray(int[] array) {
        // Visibility guaranteed by final field guarantees
        this.array = array.clone();
    }
  • 前者通过数组的长度,新建一个数组。
  • 后者通过克隆一个传入数组。也算是新建数组了。

常用操作

java">    public final int length() {
        return array.length;
    }

    public final int get(int i) {
        return getRaw(checkedByteOffset(i));
    }

    private int getRaw(long offset) {
        return unsafe.getIntVolatile(array, offset);
    }

    public final void set(int i, int newValue) {
        unsafe.putIntVolatile(array, checkedByteOffset(i), newValue);
    }

    public final void lazySet(int i, int newValue) {
        unsafe.putOrderedInt(array, checkedByteOffset(i), newValue);
    }
  • 由于array成员不是volatile的,所以get/set时,都借助了unsafe来达到volatile语义的get/set
  • lazySet则不保证可见性。
java">    public final int getAndSet(int i, int newValue) {
        return unsafe.getAndSetInt(array, checkedByteOffset(i), newValue);
    }

//Unsafe.java
    public final int getAndSetInt(Object o, long offset, int newValue) {
        int v;
        do {
            v = getIntVolatile(o, offset);
        } while (!compareAndSwapInt(o, offset, v, newValue));
        return v;
    }
  • getAndSet直接设置新值,但getAndSetInt也是循环加CAS,因为需要正确返回设置成功时的旧值。
  • getAndSet顾名思义,获得旧值,设置为新值。
java">    public final boolean compareAndSet(int i, int expect, int update) {
        return compareAndSetRaw(checkedByteOffset(i), expect, update);
    }

    private boolean compareAndSetRaw(long offset, int expect, int update) {
        return unsafe.compareAndSwapInt(array, offset, expect, update);
    }
  • 最终使用unsafe.compareAndSwapInt,反正将array当成普通对象,直接从地址偏移来进行CAS操作。
java">    public final int getAndIncrement(int i) {
        return getAndAdd(i, 1);  //获得旧值,所以直接结果就行
    }

    public final int getAndDecrement(int i) {
        return getAndAdd(i, -1);
    }

    public final int getAndAdd(int i, int delta) {  //此函数被复用
        return unsafe.getAndAddInt(array, checkedByteOffset(i), delta);
    }

    public final int incrementAndGet(int i) {
        return getAndAdd(i, 1) + 1;  //获得新值,所以需要加1
    }

    public final int decrementAndGet(int i) {
        return getAndAdd(i, -1) - 1;  //获得新值,所以需要减1
    }

    public final int addAndGet(int i, int delta) {
        return getAndAdd(i, delta) + delta;  //获得新值,所以需要加delta
    }

以上就是一些常用操作。

java">    public final int getAndUpdate(int i, IntUnaryOperator updateFunction) {
        long offset = checkedByteOffset(i);
        int prev, next;
        do {
            prev = getRaw(offset);  //Volatile语义地获得i索引的值
            next = updateFunction.applyAsInt(prev);  //调用int的一元方程
        } while (!compareAndSetRaw(offset, prev, next));
        return prev;//返回旧值
    }

    public final int updateAndGet(int i, IntUnaryOperator updateFunction) {
        long offset = checkedByteOffset(i);
        int prev, next;
        do {
            prev = getRaw(offset);
            next = updateFunction.applyAsInt(prev);
        } while (!compareAndSetRaw(offset, prev, next));
        return next;//返回新值
    }

    public final int getAndAccumulate(int i, int x,
                                      IntBinaryOperator accumulatorFunction) {
        long offset = checkedByteOffset(i);
        int prev, next;
        do {
            prev = getRaw(offset);
            next = accumulatorFunction.applyAsInt(prev, x);  //调用int的二元方程
        } while (!compareAndSetRaw(offset, prev, next));
        return prev;
    }

    public final int accumulateAndGet(int i, int x,
                                      IntBinaryOperator accumulatorFunction) {
        long offset = checkedByteOffset(i);
        int prev, next;
        do {
            prev = getRaw(offset);
            next = accumulatorFunction.applyAsInt(prev, x);
        } while (!compareAndSetRaw(offset, prev, next));
        return next;
    }

以上配合两种函数式接口,实现了些常用操作。

java">    public String toString() {
        int iMax = array.length - 1;
        if (iMax == -1)
            return "[]";

        StringBuilder b = new StringBuilder();
        b.append('[');
        for (int i = 0; ; i++) {
            b.append(getRaw(byteOffset(i)));//注意,也Volatile语义地获得i索引的值
            if (i == iMax)
                return b.append(']').toString();
            b.append(',').append(' ');
        }
    }

总结

  • 相比其他原子类,原子数组类AtomicIntegerArray,它的成员不再是volatile的,而是final的。从设计上来讲,作为原子数组类的数组成员初始化后,确实也不应该被修改引用了。
  • 计算数组元素的地址偏移时,需要两个值,base和元素索引。
  • get/set函数不能像其他原子类一样,get直接return成员,set直接赋值成员。只能借助unsafe的getIntVolatile / putIntVolatile

http://www.niftyadmin.cn/n/768206.html

相关文章

AQS深入理解 hasQueuedPredecessors源码分析 JDK8

文章目录前言流程hasQueuedPredecessors分析为什么先读取tail&#xff0c;再读取head先读取tail&#xff0c;再读取head的好处是否需要考虑 指令重排序虚假返回的true和false虚假返回true虚假返回false前言 Queries whether any threads have been waiting to acquire longer t…

Java并发 volatile可见性的验证

文章目录普通读 无法及时获得 主内存变量volatile读 及时获得 主内存变量普通读sleep普通读同步块同步块 遭遇 锁消除普通读System.out.println总结普通读 无法及时获得 主内存变量 public class volatileTest {static boolean flag false;//非volatile变量public static voi…

AQS深入理解 setHeadAndPropagate源码分析 JDK8

文章目录前言共享锁获取流程setHeadAndPropagate分析总结前言 Sets head of queue, and checks if successor may be waiting in shared mode, if so propagating if either propagate > 0 or PROPAGATE status was set. 此函数被共享锁操作而使用。这个函数用来将传入参数设…

AQS深入理解 doReleaseShared源码分析 JDK8

文章目录前言调用doReleaseShared的流程doReleaseShared分析head状态为0的情况特殊情况PROPAGATE状态设置后&#xff0c;并没有被检测到总结前言 Release action for shared mode – signals successor and ensures propagation. (Note: For exclusive mode, release just amou…

AQS深入理解 shouldParkAfterFailedAcquire源码分析 状态为0或PROPAGATE的情况分析

文章目录流程分析谁是调用shouldParkAfterFailedAcquire的线程什么时候会从parkAndCheckInterrupt中唤醒释放锁后&#xff0c;唤醒head后继的条件shouldParkAfterFailedAcquire源码分析关于shouldParkAfterFailedAcquire的疑问为什么检测到0后&#xff0c;一定要设置成SIGNAL&a…

AQS深入理解系列(一) 独占锁的获取过程

文章目录前言AQS实现核心state等待队列CAS根据实现核心找AQS的成员state等待队列CAS观察ReentrantLock的内部类公平的、不响应中断的 独占锁的获取tryAcquireaddWaiterenqenq的尾分叉 与 prev的有效性acquireQueuedshouldParkAfterFailedAcquireparkAndCheckInterrupt不公平的、…

AQS深入理解系列(二) 独占锁的释放过程

文章目录前言ReentrantLock.unlock()releasetryReleaseunparkSuccessorhead后继从哪里被唤醒前言 在上一篇文章AQS深入理解 独占锁的获取过程中&#xff0c;我们分析了独占锁的获取过程。相对于获取过程&#xff0c;独占锁的释放过程则相对简单了很多。 Lock lock new Reent…

AQS深入理解系列(三)共享锁的获取与释放

文章目录前言共享锁与独占锁的区别观察Semaphore的内部类共享锁的获取共享锁的释放head状态为0的情况同时执行doReleaseShared总结前言 在前面两篇系列文章中&#xff0c;已经讲解了独占锁的获取和释放过程&#xff0c;而共享锁的获取与释放过程也很类似&#xff0c;如果你前面…