看图学源码之 Atomic 类源码浅析二(cas + 分治思想的原子累加器)

news2025/1/18 20:08:06

原子累加器

相较于上一节看图学源码 之 Atomic 类源码浅析一(cas + 自旋操作的 AtomicXXX原子类)说的的原子类,原子累加器的效率会更高

在这里插入图片描述

XXXXAdderXXXAccumulator 区别就是 Adder只有add 方法,Accumulator是可以进行自定义运算方法的

始于 Striped64
abstract class Striped64 extends Number {

  // cpu 运行核数, 控制数组的大小
   static final int NCPU = Runtime.getRuntime().availableProcessors();
  
  // 当非空时,大小是 2 的幂。 
  transient volatile Cell[] cells;
  
  // 表初始化竞争期间的后备值  通过  CAS 更新  就是 value
  transient volatile long base;
  
  // 锁的 标志位 调整单元大小和/或创建单元时使用自旋锁(通过 CAS 锁定)
  transient volatile int cellsBusy;
  
  
    //Base的cas 操作
    final boolean casBase(long cmp, long val) {
        return UNSAFE.compareAndSwapLong(this, BASE, cmp, val);
    }

    // CellsBusy的cas操作
    final boolean casCellsBusy() {
        return UNSAFE.compareAndSwapInt(this, CELLSBUSY, 0, 1);
    }

  
// 主要是给不同的线程找到数组中不同的下标
    // 当前线程的探测值
    static final int getProbe() {
        return UNSAFE.getInt(Thread.currentThread(), PROBE);
    }
    // 给定线程的给定探测值
    static final int advanceProbe(int probe) {
        probe ^= probe << 13;   // xorshift
        probe ^= probe >>> 17;
        probe ^= probe << 5;
        UNSAFE.putInt(Thread.currentThread(), PROBE, probe);
        return probe;
    }
  ...
  
  ...
    // Unsafe mechanics
    // Unsafe 的获取 和 偏移量的获取
    private static final sun.misc.Unsafe UNSAFE;
    private static final long BASE;
    private static final long CELLSBUSY;
    private static final long PROBE;
    static {
        try {
            UNSAFE = sun.misc.Unsafe.getUnsafe();
            Class<?> sk = Striped64.class;
            BASE = UNSAFE.objectFieldOffset(sk.getDeclaredField("base"));
            CELLSBUSY = UNSAFE.objectFieldOffset(sk.getDeclaredField("cellsBusy"));
            Class<?> tk = Thread.class;
            PROBE = UNSAFE.objectFieldOffset(tk.getDeclaredField("threadLocalRandomProbe"));
        } catch (Exception e) {
            throw new Error(e);
        }
    }

}
内部类

@sun.misc.Contended——解决伪共享,进行字节填充

@sun.misc.Contended static final class Cell {
  // 操作的数
    volatile long value;
  
  // 构造器
    Cell(long x) { value = x; }
  
  // 进行 cas 
    final boolean cas(long cmp, long val) {
        return UNSAFE.compareAndSwapLong(this, valueOffset, cmp, val);
    }
}
LongAdder

可能会出现一边正在进行 累加操作,一边又在执行求和操作,所以就导致了不是 强一致性,而是最终一致性

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

public class LongAdder extends Striped64 implements Serializable {
    private static final long serialVersionUID = 7249069246863182397L;

    public LongAdder() {
    }


    public void add(long x) {
        Cell[] as; long b, v; int m; Cell a;
      // 数组是不是 null(判断有没有发生竞争,因为只有竞争发生才会初始化数组)
      // 没有初始化(就是没有竞争) 直接对 base 的值+x 失败
        if ((as = cells) != null || !casBase(b = base, b + x)) {
          //有竞争的时候
            boolean uncontended = true;
          // 数组还是没有初始化 || 数组初始化,但是数组的长度  < 0 || 数组中的该位置的值是 null (表示这个下标没有初始化)|| cas的方式把当前位置的值 + x ,cas 失败
            if (as == null || (m = as.length - 1) < 0 ||
                (a = as[getProbe() & m]) == null ||
                !(uncontended = a.cas(v = a.value, v + x)))
              // 发生冲突都会走这里
                longAccumulate(x, null, uncontended);
        }
    }

  
    public void increment() {
        add(1L);
    }

    public void decrement() {
        add(-1L);
    }

  // 返回当前总和。返回的值不是原子快照;
  // 在没有并发更新的情况下调用会返回准确的结果,但计算总和时发生的并发更新可能不会被合并
	// 所以不是 强一致性的
    public long sum() {
        Cell[] as = cells; Cell a;
        long sum = base;
      //数组不是 null
        if (as != null) {
          //遍历数组,
            for (int i = 0; i < as.length; ++i) {
              //数组中的槽位不是 null,对槽位的数据进行运算,赋值加到base中
                if ((a = as[i]) != null)
                    sum += a.value;
            }
        }
      //返回总的值
        return sum;
    }

 // 将保持总和为零的变量重置。此方法可能是创建新加法器的有用替代方法,但仅在没有并发更新时才有效。
 // 由于此方法本质上是活泼的,因此仅应在  已知没有线程同时更新时才使用它。
    public void reset() {
        Cell[] as = cells; Cell a;
        base = 0L;
        if (as != null) {
            // 数组存在,遍历数组,将数组中所有的值设置为 0 
            for (int i = 0; i < as.length; ++i) {
                if ((a = as[i]) != null)
                    a.value = 0L;
            }
        }
    }

    // 相当于sum后跟reset 
    // 该方法可以应用于例如 多线程计算之间的静止点期间。
    // 如果此方法同时有更新,则不能保证返回值是重置之前发生的最终值。
    public long sumThenReset() {
        Cell[] as = cells; Cell a;
        long sum = base;
        base = 0L;
        if (as != null) {
          // 数组存在,遍历数组,先求和  后把数组中所有的值设置为 0 
            for (int i = 0; i < as.length; ++i) {
                if ((a = as[i]) != null) {
                    sum += a.value;
                    a.value = 0L;
                }
            }
        }
        return sum;
    }


  // 返回sum的字符串表示形式
    public String toString() {
        return Long.toString(sum());
    }


   // 返回sum
    public long longValue() {
        return sum();
    }

  //缩小基元转换后以 int 形式返回sum 
    public int intValue() {
        return (int)sum();
    }

  // 加宽基元转换后以float形式返回sum 
    public float floatValue() {
        return (float)sum();
    }
  //加宽基元转换后以 double 形式返回sum 
    public double doubleValue() {
        return (double)sum();
    }

    private static class SerializationProxy implements Serializable {
        private static final long serialVersionUID = 7249069246863182397L;

        private final long value;

      // sum() 返回的当前值。
        SerializationProxy(LongAdder a) {
            value = a.sum();
        }
				// 返回一个LongAdder对象,其初始状态由该代理保存。
        private Object readResolve() {
            LongAdder a = new LongAdder();
            a.base = value;
            return a;
        }
    }

    private Object writeReplace() {
        return new SerializationProxy(this);
    }
  
    private void readObject(java.io.ObjectInputStream s)
        throws java.io.InvalidObjectException {
        throw new java.io.InvalidObjectException("Proxy required");
    }

}
striped64中的 longAccumulate

在这里插入图片描述

final void longAccumulate(long x, LongBinaryOperator fn,boolean wasUncontended) {
    int h;
   // 拿 hash 值,拿不到强制获取
    if ((h = getProbe()) == 0) {
        ThreadLocalRandom.current(); // force initialization
        h = getProbe();
      // 将 wasUncontended 的值设为 true,表示当前线程是未争用的。
        wasUncontended = true;
    }
    boolean collide = false;                // True if last slot nonempty
    for (;;) {
        Cell[] as; Cell a; int n; long v;
   // 分支 1
      //数组已经初始化,出现了竞争
        if ((as = cells) != null && (n = as.length) > 0) {
        // 分支1.1
          		// 当前位置的值是null
            if ((a = as[(n - 1) & h]) == null) {
              // 锁的标志位 == 0 ,没有加锁 
                if (cellsBusy == 0) {       // Try to attach new Cell
                    Cell r = new Cell(x);   // Optimistically create
                   // 加锁
                    if (cellsBusy == 0 && casCellsBusy()) {
                        boolean created = false;   
                        try {               // Recheck under lock
                            Cell[] rs; int m, j;
                          //  加锁之后再次检查指定位置是否为空 
                          //  数组初始化过了 && 当前位置的值不是null
                            if ((rs = cells) != null &&
                                (m = rs.length) > 0 &&
                                rs[j = (m - 1) & h] == null) {
                              // 给数组的指定位置设置为  之前设置过的cell对象
                                rs[j] = r;
                              // 创建成功
                                created = true;
                            }
                        } finally {
                          // 解锁
                            cellsBusy = 0;
                        }
                        if (created)
                            break;
                        continue;           // Slot is now non-empty
                    }
                }
              // 有人加锁了,发生了冲突 避免在当前位置发生碰撞的情况下继续进行操作,将 collide 标志位设置为 false。
                collide = false;
            }
          // 分支1.2
          	 // 没有发生竞争
            else if (!wasUncontended)       // CAS already known to fail
              // 此时是发生了竞争
                wasUncontended = true;      // Continue after rehash
          // 分支1.3
          		// cas 的方式更新 此位置的值, cas 失败表示有线程正在此位置执行操作
            else if (a.cas(v = a.value, ((fn == null) ? v + x :fn.applyAsLong(v, x))))
                break;
          // 分支1.4
              // n > cpu 的个数  当前分段数组的长度是否已经达到或超过了处理器的数量。
              //如果是,说明分段数组已经达到了最大的容量或者已经很大了,不再需要继续进行扩容操作。
           		
              // 或者 cells  发生了变化,当前线程获取到的分段数组引用是否与共享变量中的引用相等。
              // 如果不相等,说明在当前线程获取到分段数组的过程中,有其他线程进行了修改,即分段数组已经发生了变化。
            else if (n >= NCPU || cells != as)
                collide = false;            // At max size or stale
          // 分支1.5
           // 此时是发生了碰撞的 collide 被设置为 true 
            else if (!collide)
                collide = true;
          // 分支1.6 
           			// 扩容
          							// 没有被锁 && cas  的方式 成功加锁 
            else if (cellsBusy == 0 && casCellsBusy()) {
                try {
                  			// 数组没有变化
                    if (cells == as) {      // Expand table unless stale
                       // as 数组长度扩大一倍
                        Cell[] rs = new Cell[n << 1];
                     	// 元素直接赋值
                        for (int i = 0; i < n; ++i)
                            rs[i] = as[i];
                        cells = rs;
                    }
                } finally {
                   // 解锁
                    cellsBusy = 0;
                }
              //没有冲突
                collide = false;
              //扩容成功,继续循环
                continue;                   // Retry with expanded table
            }
           // 更新 hash 值
            h = advanceProbe(h);
        }
// 分支 2
      // 此处数组没有进行初始化,此时进行初始化
      // 锁的标志为 == 0  &&  数组没有改变(多线程情况下该线程没有被其他线程初始化)  && cas 成功的把锁的标志位 设置为 1(加锁流程)
      // 当前的 cells 数组没有被其他线程占用,并且成功获取了 cellsBusy 锁
        else if (cellsBusy == 0 && cells == as && casCellsBusy()) {
            boolean init = false;
            try {                           // Initialize table
              // 加完锁之后再次判断一次  cells 数组没有发生过变化
                if (cells == as) { 
                  // 数组 长度默认为2
                    Cell[] rs = new Cell[2];
                  // 给rs 赋值为 要加入的 x
                    rs[h & 1] = new Cell(x);
                  // 将 cells 数组变更为 rs
                    cells = rs;
                  // 初始化成功
                    init = true;
                }
            } finally {
              // 解锁
                cellsBusy = 0;
            }
            if (init)    //初始化成功
                break;    // 退出循环
        }
// 分支 3  cas 的方式 操作 base  , fn 函数式接口的方法  == null 默认加法,否则就是定义的方法
        else if (casBase(v = base, ((fn == null) ? v + x : fn.applyAsLong(v, x))))
            break;                          // Fall back on using base  退出循环
    }
}
LongAccumulator

在这里插入图片描述

在这里插入图片描述

image-20231207001843693

public class LongAccumulator extends Striped64 implements Serializable {
    private static final long serialVersionUID = 7249069246863182397L;

    private final LongBinaryOperator function;
    private final long identity;

    public LongAccumulator(LongBinaryOperator accumulatorFunction,
                           long identity) {
        this.function = accumulatorFunction;
        base = this.identity = identity;
    }

    // 更新值
    public void accumulate(long x) {
        Cell[] as; long b, v, r; int m; Cell a;
      // 有竞争  || ( cas 运算 base 的值成功  && 对 base进行cas更新失败 )
        if ((as = cells) != null ||
            //  function.applyAsLong  函数式接口 
            (r = function.applyAsLong(b = base, x)) != b && !casBase(b, r)) {
            boolean uncontended = true;
					// 出现了竞争
          // 数组还是没有初始化 || 数组初始化,但数组的长度 < 0 || 数组中的该位置的值是 null (表示这个下标没有初始化)|| (cas的方式运算当前位置的值 失败 && cas 更新当前位置的值也失败)
            if (as == null || (m = as.length - 1) < 0 ||
                (a = as[getProbe() & m]) == null ||
                !(uncontended =
                  (r = function.applyAsLong(v = a.value, x)) == v ||
                  a.cas(v, r)))
                longAccumulate(x, function, uncontended);
        }
    }

    // 返回当前值。返回的值不是原子快照;
    // 在没有并发更新的情况下调用会返回准确的结果,但计算值时发生的并发更新可能不会被合并。
     
    public long get() {
        Cell[] as = cells; Cell a;
        long result = base;
       // 数组存在
        if (as != null) {
          // 遍历数组
            for (int i = 0; i < as.length; ++i) {
              //数组中的槽位不是 null
                if ((a = as[i]) != null)
                  //对槽位的数据进行运算,赋值加到base中
                    result = function.applyAsLong(result, a.value);
            }
        }
      // 并返回总的值
        return result;
    }

    // 重置变量以维护对标识值的更新。
  	// 此方法可能是创建新更新程序的有用替代方法,但仅在没有并发更新时才有效。
    // 由于此方法本质上是活泼的,因此仅应在已知没有线程同时更新时才使用它。 
    public void reset() {
        Cell[] as = cells; Cell a;
        base = identity;
       // 数组存在
        if (as != null) {
          // 遍历数组
            for (int i = 0; i < as.length; ++i) {
              //数组中的槽位不是 null
                if ((a = as[i]) != null)
                  //将槽位的值设置为  identity
                    a.value = identity;
            }
        }
    }

  // 效果相当于get后面跟着reset 。
  // 该方法可以应用于例如多线程计算之间的静止点期间。
  // 如果此方法同时有更新,则不能保证返回值是重置之前发生的最终值。
    public long getThenReset() {
        Cell[] as = cells; Cell a;
        long result = base;
        base = identity;
      // 数组存在
        if (as != null) {
          // 遍历数组
            for (int i = 0; i < as.length; ++i) {
              //数组中的槽位不是 null
                if ((a = as[i]) != null) {
                  // 将槽位的值设置为  identity
                  // 对槽位的数据进行运算,赋值加到base中
                    long v = a.value;
                    a.value = identity;
                    result = function.applyAsLong(result, v);
                }
            }
        }
        return result;
    }


    public String toString() {
        return Long.toString(get());
    }


    public long longValue() {
        return get();
    }


    public int intValue() {
        return (int)get();
    }


    public float floatValue() {
        return (float)get();
    }


    public double doubleValue() {
        return (double)get();
    }

    /**
     * 序列化代理,用于避免以序列化形式引用非公共 Striped64 超类
     */
    private static class SerializationProxy implements Serializable {
        private static final long serialVersionUID = 7249069246863182397L;

        private final long value;

        private final LongBinaryOperator function;

        private final long identity;

        SerializationProxy(LongAccumulator a) {
            function = a.function;
            identity = a.identity;
            value = a.get();
        }


        private Object readResolve() {
            LongAccumulator a = new LongAccumulator(function, identity);
            a.base = value;
            return a;
        }
    }


    private Object writeReplace() {
        return new SerializationProxy(this);
    }

    private void readObject(java.io.ObjectInputStream s)
        throws java.io.InvalidObjectException {
        throw new java.io.InvalidObjectException("Proxy required");
    }

}

在Double 中会有 doubleToRawLongBits的操作,主要是检查数组越界的

DoubleAdder

在这里插入图片描述

public class DoubleAdder extends Striped64 implements Serializable {
    private static final long serialVersionUID = 7249069246863182397L;

    /*
请注意,我们必须使用“long”作为底层表示,因为 double 没有compareAndSet,因为任何 CAS 实现中使用的按位等于与双精度等于不同
然而,我们仅使用 CAS 来检测和缓解争用,无论如何,按位等于效果最好。
原则上,这里使用的 longdouble 转换在大多数平台上基本上应该是免费的,因为它们只是重新解释位。
     */


    public DoubleAdder() {
    }


    public void add(double x) {
        Cell[] as; long b, v; int m; Cell a;
      // 数组存在 || 对 base 进行 cas运算操作失败  
if ((as = cells) != null ||!casBase(b = base,Double.doubleToRawLongBits(Double.longBitsToDouble(b) + x))) {
   // 
            boolean uncontended = true;
  // 数组为 空 || 数组的长度 < 0  || 当前位置的值为 null  ||  对该位置的值进行cas 运算失败
            if (as == null || (m = as.length - 1) < 0 ||
                (a = as[getProbe() & m]) == null ||
                !(uncontended = a.cas(v = a.value,
                                      Double.doubleToRawLongBits
                                      (Double.longBitsToDouble(v) + x))))
                doubleAccumulate(x, null, uncontended);
        }
    }

    /**
    返回当前总和。
    返回的值不是原子快照;在没有并发更新的情况下调用会返回准确的结果,但计算总和时发生的并发更新可能不会被合并。
    由于浮点算术不是严格关联的,因此返回的结果不需要与在单个变量的一系列连续更新中获得的值相同。
     */
    public double sum() {
        Cell[] as = cells; Cell a;
        double sum = Double.longBitsToDouble(base);
        if (as != null) {
            for (int i = 0; i < as.length; ++i) {
                if ((a = as[i]) != null)
                    sum += Double.longBitsToDouble(a.value);
            }
        }
        return sum;
    }

    /**
    将保持总和为零的变量重置。
    此方法可能是创建新加法器的有用替代方法,但仅在没有并发更新时才有效。
    由于此方法本质上是活泼的,因此仅应在已知没有线程同时更新时才使用它。
     */
    public void reset() {
        Cell[] as = cells; Cell a;
        base = 0L; // relies on fact that double 0 must have same rep as long
        if (as != null) {
            for (int i = 0; i < as.length; ++i) {
                if ((a = as[i]) != null)
                    a.value = 0L;
            }
        }
    }

    /**
     相当于sum后跟reset 。
     该方法可以应用于例如多线程计算之间的静止点期间。如果此方法同时有更新,则不能保证返回值是重置之前发生的最终值。
     */
    public double sumThenReset() {
        Cell[] as = cells; Cell a;
        double sum = Double.longBitsToDouble(base);
        base = 0L;
        if (as != null) {
            for (int i = 0; i < as.length; ++i) {
                if ((a = as[i]) != null) {
                    long v = a.value;
                    a.value = 0L;
                    sum += Double.longBitsToDouble(v);
                }
            }
        }
        return sum;
    }

   
    public String toString() {
        return Double.toString(sum());
    }

  
    public double doubleValue() {
        return sum();
    }

   
    public long longValue() {
        return (long)sum();
    }

   
    public int intValue() {
        return (int)sum();
    }

   
    public float floatValue() {
        return (float)sum();
    }

  
    private static class SerializationProxy implements Serializable {
        private static final long serialVersionUID = 7249069246863182397L;

       
        private final double value;

        SerializationProxy(DoubleAdder a) {
            value = a.sum();
        }

        private Object readResolve() {
            DoubleAdder a = new DoubleAdder();
            a.base = Double.doubleToRawLongBits(value);
            return a;
        }
    }

    private Object writeReplace() {
        return new SerializationProxy(this);
    }

    private void readObject(java.io.ObjectInputStream s)
        throws java.io.InvalidObjectException {
        throw new java.io.InvalidObjectException("Proxy required");
    }

}
striped64中的 doubleAccumulate

和上面的 striped64中的 longAccumulate 几乎一模一样,只有doubleToRawLongBits部分的细微差别

final void doubleAccumulate(double x, DoubleBinaryOperator fn,boolean wasUncontended) {
    int h;
     // 拿 hash 值,拿不到强制获取
    if ((h = getProbe()) == 0) {
        ThreadLocalRandom.current(); // force initialization
        h = getProbe();
        wasUncontended = true;
    }
    boolean collide = false;                // True if last slot nonempty
    for (;;) {
        Cell[] as; Cell a; int n; long v;
// 分支 1
      //数组已经初始化,出现了竞争
        if ((as = cells) != null && (n = as.length) > 0) {
          // 分支1.1
          		// 当前位置的值是null
            if ((a = as[(n - 1) & h]) == null) {
                // 锁的标志位 == 0 ,没有加锁 
                if (cellsBusy == 0) {       // Try to attach new Cell
                    Cell r = new Cell(Double.doubleToRawLongBits(x));
                    // 加锁
                    if (cellsBusy == 0 && casCellsBusy()) {
                        boolean created = false;
                        try {               // Recheck under lock
                            Cell[] rs; int m, j;
                        // 数组初始化过了 && 当前位置的值不是null
                            if ((rs = cells) != null &&
                                (m = rs.length) > 0 &&
                                rs[j = (m - 1) & h] == null) {
                                  // 给数组的位置设置为  之前设置过的cell对象
                                rs[j] = r;
                                // 创建成功
                                created = true;
                            }
                        } finally {
                            // 解锁
                            cellsBusy = 0;
                        }
                        if (created)
                            break;
                        continue;           // Slot is now non-empty
                    }
                }
                // 有人加锁了,发生了冲突
                collide = false;
            }
            // 分支1.2
          	 // 没有发生竞争
            else if (!wasUncontended)       // CAS already known to fail
                 // 此时是发生了竞争
                wasUncontended = true;      // Continue after rehash
           // 分支1.3
          		// cas 的方式更新 此位置的值, cas 失败表示有线程正在此位置执行操作
            else if (a.cas(v = a.value,
                           ((fn == null) ?
                            Double.doubleToRawLongBits
                            (Double.longBitsToDouble(v) + x) :
                            Double.doubleToRawLongBits
                            (fn.applyAsDouble
                             (Double.longBitsToDouble(v), x)))))
                break;
          // 分支1.4
              // n > cpu 的个数 或者 cells  发生了变化,表示 之前没有发生碰撞,不能扩容
            else if (n >= NCPU || cells != as)
                collide = false;            // At max size or stale
         // 分支1.5
           // 此时是发生了碰撞的 collide 被设置为 true 
            else if (!collide)
                collide = true;
             // 分支1.6 
           			// 扩容
          							// 没有被锁 && 成功加锁 
            else if (cellsBusy == 0 && casCellsBusy()) {
                try {
                  // 数组没有变化
                    if (cells == as) {      // Expand table unless stale
                      // as 数组长度扩大一倍
                        Cell[] rs = new Cell[n << 1];
                      // 元素直接赋值
                        for (int i = 0; i < n; ++i)
                            rs[i] = as[i];
                        cells = rs;
                    }
                } finally {
                    // 解锁
                    cellsBusy = 0;
                }
                 //没有冲突
                collide = false;
                 //扩容成功,继续循环
                continue;                   // Retry with expanded table
            }
          // 获取hash 值
            h = advanceProbe(h);
        }
// 分支 2
      // 此处数组没有进行初始化,此时进行初始化
      // 锁的标志为 == 0  &&  数组没有改变(多线程情况下该线程没有被其他线程初始化)  && cas 成功的把锁的标志位 设置为 1(枷锁流程)
        else if (cellsBusy == 0 && cells == as && casCellsBusy()) {
            boolean init = false;
            try {   // Initialize table
                 // 加完锁之后再次判断一次  cells 数组没有发生过变化
                if (cells == as) {
                  // 数组 长度默认为2
                    Cell[] rs = new Cell[2];
                   // 给rs 赋值为 要加入的 x
                    rs[h & 1] = new Cell(Double.doubleToRawLongBits(x));
                   // 将 cells 数组变更为 rs
                    cells = rs;
                   // 初始化成功
                    init = true;
                }
            } finally {
              // 解锁
                cellsBusy = 0;
            }
            if (init)// 初始化成功
                break;    // 退出循环
        }
// 分支 3  cas 的方式 操作 base  , fn 函数式接口的方法  == null 默认加法,否则就是定义的方法
        else if (casBase(v = base,
                         ((fn == null) ?
                          Double.doubleToRawLongBits
                          (Double.longBitsToDouble(v) + x) :
                          Double.doubleToRawLongBits
                          (fn.applyAsDouble
                           (Double.longBitsToDouble(v), x)))))
            break;                          // Fall back on using base// 退出循环
    }
}
DoubleAccumulator

在这里插入图片描述

public class DoubleAccumulator extends Striped64 implements Serializable {
    private static final long serialVersionUID = 7249069246863182397L;

    private final DoubleBinaryOperator function;
    private final long identity; // use long representation


    public DoubleAccumulator(DoubleBinaryOperator accumulatorFunction,double identity) {
        this.function = accumulatorFunction;
        base = this.identity = Double.doubleToRawLongBits(identity);
    }

  
 
    public void accumulate(double x) {
        Cell[] as; long b, v, r; int m; Cell a;
        // 数组存在 || (对 base 进行 cas的 运算操作成功  && 对base 进行cas 更新操作失败  )
        if ((as = cells) != null ||(r = Double.doubleToRawLongBits(function.applyAsDouble (Double.longBitsToDouble(b = base), x))) != b  && !casBase(b, r)) {
            boolean uncontended = true;
// 数组为 空 || 数组被初始化但是 数组的长度 < 0  || 当前位置的值为 null  ||  (对该位置的值进行cas 运算失败  || 对该值进行cas 更新失败)
            if (as == null || (m = as.length - 1) < 0 ||
                (a = as[getProbe() & m]) == null ||
                !(uncontended =(r = Double.doubleToRawLongBits(function.applyAsDouble(Double.longBitsToDouble(v = a.value), x))) == v ||a.cas(v, r)))
                doubleAccumulate(x, function, uncontended);
        }
    }


    public double get() {
        Cell[] as = cells; Cell a;
        double result = Double.longBitsToDouble(base);
        if (as != null) {
            for (int i = 0; i < as.length; ++i) {
                if ((a = as[i]) != null)
                    result = function.applyAsDouble
                        (result, Double.longBitsToDouble(a.value));
            }
        }
        return result;
    }


    public void reset() {
        Cell[] as = cells; Cell a;
        base = identity;
        if (as != null) {
            for (int i = 0; i < as.length; ++i) {
                if ((a = as[i]) != null)
                    a.value = identity;
            }
        }
    }


    public double getThenReset() {
        Cell[] as = cells; Cell a;
        double result = Double.longBitsToDouble(base);
        base = identity;
        if (as != null) {
            for (int i = 0; i < as.length; ++i) {
                if ((a = as[i]) != null) {
                    double v = Double.longBitsToDouble(a.value);
                    a.value = identity;
                    result = function.applyAsDouble(result, v);
                }
            }
        }
        return result;
    }


    public String toString() {
        return Double.toString(get());
    }

    public double doubleValue() {
        return get();
    }


    public long longValue() {
        return (long)get();
    }

    public int intValue() {
        return (int)get();
    }

    public float floatValue() {
        return (float)get();
    }

    private static class SerializationProxy implements Serializable {
        private static final long serialVersionUID = 7249069246863182397L;

        private final double value;

        private final DoubleBinaryOperator function;

        private final long identity;

        SerializationProxy(DoubleAccumulator a) {
            function = a.function;
            identity = a.identity;
            value = a.get();
        }


        private Object readResolve() {
            double d = Double.longBitsToDouble(identity);
            DoubleAccumulator a = new DoubleAccumulator(function, d);
            a.base = Double.doubleToRawLongBits(value);
            return a;
        }
    }


    private Object writeReplace() {
        return new SerializationProxy(this);
    }


    private void readObject(java.io.ObjectInputStream s)
        throws java.io.InvalidObjectException {
        throw new java.io.InvalidObjectException("Proxy required");
    }

}

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

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

相关文章

Redis如何做内存优化?

Redis如何做内存优化&#xff1f; 1、缩短键值的长度 缩短值的长度才是关键&#xff0c;如果值是一个大的业务对象&#xff0c;可以将对象序列化成二进制数组&#xff1b; 首先应该在业务上进行精简&#xff0c;去掉不必要的属性&#xff0c;避免存储一些没用的数据&#xff1…

idea__SpringBoot微服务07——Thymeleaf

Thymeleaf 一、Thymeleaf入门二、Thymeleaf语法————————创作不易&#xff0c;如觉不错&#xff0c;随手点赞&#xff0c;关注&#xff0c;收藏(*&#xffe3;︶&#xffe3;)&#xff0c;谢谢~~ 新依赖&#xff1a;Thymeleaf的 <!--thymeleaf--> <dependency…

uniapp生成条形码?

首先先在插件市场找到条形码 链接&#xff1a;https://ext.dcloud.net.cn/search?q 下载到项目里面 三、在pages.json中写入以下&#xff1a; 四、在html页面 这样就已经差不多了 条形码可以出来了

java学习part41泛型

164-泛型-泛型的理解及其在集合、比较器中的使用_哔哩哔哩_bilibili 1.泛型 泛型可以加&#xff0c;加了不用也可以 2.例子 2.1List泛型 2.2Map泛型 新特性var 类似c的auto&#xff0c;可以根据后面赋值推断类型&#xff0c;在这种类型很复杂的时候可以提高可读性 3.自定义泛型…

人工智能AIGC培训讲师叶梓介绍及AI强化学习培训提纲

叶梓&#xff0c;上海交通大学计算机专业博士毕业&#xff0c;高级工程师。主研方向&#xff1a;数据挖掘、机器学习、人工智能。历任国内知名上市IT企业的AI技术总监、资深技术专家&#xff0c;市级行业大数据平台技术负责人。个人主页&#xff1a;大数据人工智能AI培训讲师叶…

git stash 对当前分支修改的内容进行暂存

我们在开发的时候往往会遇到这种情况, 在一个分支开发,写了不少内容,但是突然来了一个紧急的需求需要切换分支,去做这个需求,但是当前的分支又因为没有开发完成,不想形成一条无效的commit记录,这时我们就到暂存上场了 git stash 暂存 // 切分支之前 对当前分支修改的内容进行暂…

【算法通关村】链表反转经典问题解析

&#x1f6a9;本文已收录至算法学习之旅 一.基础反转 我们通常有两种方法反转链表&#xff0c;一种是直接操作链表实现反转操作&#xff0c;一种是建立虚拟头节点辅助实现反转操作。 力扣习题链接&#xff1a;206. 反转链表 (1) 直接操作实现反转 我们需要一个变量pre来保…

13. MySQL 日志

目录 错误日志 binlog日志 概述 日志格式 查询日志 慢查询日志 错误日志 错误日志是MySQL中最重要的日志之一&#xff0c;它记录了当mysqld启动和停止时&#xff0c;以及服务器在运行过程中发生任何严重错误时的相关信息。当数据库出现任何故障导致无法正常使用时&#…

51单片机独立按键以及矩阵按键的使用以及其原理--独立按键 K1 控制 D1 指示灯亮灭以及数码管显示矩阵按键 S1-S16 按下后键值 0-F

IO 的使用–按键 本文主要涉及8051单片机按键的使用&#xff0c;包括独立按键以及矩阵按键的使用以及其原理&#xff0c;其中代码实例包括: 1.独立按键 K1 控制 D1 指示灯亮灭 2.通过数码管显示矩阵按键 S1-S16 按下后键值 0-F 文章目录 IO 的使用--按键一、按键消抖二、独立按…

MobaXterm成功连接到开发环境后,过一段时间会自动断开。

问题现象 MobaXterm成功连接到开发环境后&#xff0c;过一段时间会自动断开。 原因 配置MobaXterm工具时&#xff0c;没有勾选“SSH keepalive”或专业版MobaXterm工具的“Stop server after”时间设置太短。

8. MySQL 触发器

目录 概述 定义 触发器特性&#xff1a; 基础操作 创建触发器 NEW和OLD 其他操作 查看触发器 删除触发器 注意事项 概述 定义 触发器&#xff0c;就是一种特殊的存储过程。触发器和存储过程一样是一个能够完成特定功能、存储在数据库服务器上的SQL片段&#xff0c;但是触…

跨平台的文本编辑器——CudaText

CudaText 是一个轻量级、跨平台的文本编辑器&#xff0c;它免费开源&#xff0c;启动速度非常快&#xff0c;有拓展功能&#xff0c;可安装插件。 下载 浏览器搜索框输入CudaText - Home进行搜索&#xff0c; 选择官网进入&#xff0c; 进入官网界面如下&#xff1a;选择点击…

C++初阶-vector的介绍及使用

vector的介绍及使用 一、vector的介绍1.1 vector的概念 二、vector的使用2.1 vector的定义2.2 vector iterator的使用2.3 vector空间增长问题2.4 vector的增删改查2.5 vector的整体代码实现2.5.1 vector的常用内置函数使用2.5.2 vector的访问方式及测试函数 三、vector迭代器失…

15.Java程序设计-基于SSM框架的微信小程序校园求职系统的设计与实现

摘要&#xff1a; 本研究旨在设计并实现一款基于SSM框架的微信小程序校园求职系统&#xff0c;以提升校园求职流程的效率和便捷性。通过整合微信小程序平台和SSM框架的优势&#xff0c;本系统涵盖了用户管理、职位发布与搜索、简历管理、消息通知等多个功能模块&#xff0c;为…

Ubuntu与Windows通讯传输文件(FTP服务器版)(没用的方法,无法施行)

本文介绍再Windows主机上建立FTP服务器&#xff0c;并且在Ubuntu虚拟机上面访问Windows上FTP服务器的方法 只要按照上图配置就可以了 第二部&#xff1a;打开IIS管理控制台 右击网站&#xff0c;新建FTP站点。需要注意的一点是在填写IP地址的时候&#xff0c;只需要填写Window…

python操作MySQL数据库简单示例

通过python的pymysql模块&#xff0c;实现数据库表的创建、插入以及查询的简单示例。 1. 数据库表创建。 # -*- coding:utf-8 -*- """ 使用python操作MySQL数据库示例 执行sql_create语句建立数据库表books """import pymysqldef main():# 首先…

nodejs微信小程序+python+PHP北京地铁票务APP-计算机毕业设计推荐 -安卓

目 录 摘 要 I ABSTRACT II 目 录 II 第1章 绪论 1 1.1背景及意义 1 1.2 国内外研究概况 1 1.3 研究的内容 1 第2章 相关技术 3 2.1 nodejs简介 4 2.2 express框架介绍 6 2.4 MySQL数据库 4 第3章 系统分析 5 3.1 需求分析 5 3.2 系统可行性分析 5 3.2.1技术可行性&#xff1a;…

postgresql自带指令命令系列三

目录 简介 bin目录 28.pg_verifybackup 29.pg_waldump 30.postgres 31.postmaster -> postgres 32.psql 33.reindexdb 34.vacuumdb 35.vacuumlo 总结&#xff1a; 简介 在安装postgresql数据库的时候会需要设置一个关于postgresql数据库的PATH变量 export PATH/…

PTA 输出三角形面积和周长

#include<stdio.h> #include<math.h>//使用sqrt需要使用此头文件 int main() {int a, b, c, d;float s, area, perimeter;scanf("%d %d %d", &a, &b, &c);if (a b < c || a c < b || b c < a)//三角形任意两边之和大于第三边pri…

【小白专用】在 vs 中使用 nuget 安装NPOI

C#操作Excel有多种方法&#xff0c;如通过数据库的方式来读写Excel的OleDb方式&#xff0c;但是OleDb方式需要安装微软office&#xff0c;还可以通过COM组件方式操作Excel&#xff0c;也需要安装微软Excel。如果不想安装微软办公套餐可以使用ClosedXML、EPPlus、NPOI。本文主要…