原子变量 一、原子变量类 原子变量类相当于是一种泛化的 volatile 变量,能够支持原子和有条件的读写操作。
1.1 原子类型
标量类:AtomicInteger、AtomicLong、AtomicBoolean、AtomicReference
数组类:AtomicIntegerArray、AtomicLongArray、AtomicBooleanArray
更新器类:AtomicIntegerFieldUpdater、AtomicLongFieldUpdater、AtomicBooleanFieldUpdater
复合变量类:AtomicMarkableReference
1.2 性能比较
原子变量比锁的粒度更细、量级更轻
原子变量将发生竞争的范围缩小至单个变量上
原子变量通常会更快,因为它不需要挂起或重新调度线程
在高度竞争的情况下,锁的性能会比原子变量更好,原因是锁会挂起线程,避免了很多线程上下文切换
在中低竞争的情况下,原子变量能提供更高的可伸缩性
二、非阻塞算法
非阻塞的计数器:单状态
非阻塞的数字区域:单不可变对象,包含多状态
非阻塞的栈:单状态
非阻塞的队列:多状态
2.1 非阻塞的计数器 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 public class CasCounter { private SimulatedCAS value; public int getValue () { return value.get(); } public int increment () { int v; do { v = value.get(); } while (v != value.compareAndSwap(v, v + 1 )); return v + 1 ; } }
2.2 非阻塞的数字区域 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 public class CasNumberRange { private static class IntPair { final int lower; final int upper; public IntPair (int lower, int upper) { this .lower = lower; this .upper = upper; } } private final AtomicReference<IntPair> values = new AtomicReference <>(new IntPair (0 , 0 )); public int getLower () { return values.get().lower; } public int getUpper () { return values.get().upper; } public void setLower (int i) { while (true ) { IntPair oldValue = values.get(); if (i > oldValue.upper) { throw new IllegalArgumentException ("Can't set lower to " + i + " > upper" ); } IntPair newValue = new IntPair (i, oldValue.upper); if (values.compareAndSet(oldValue,newValue)) { break ; } } } public void setUpper (int i) { while (true ) { IntPair oldValue = values.get(); if (i < oldValue.lower) { throw new IllegalArgumentException ("Can't set upper to " + i + " < lower" ); } IntPair newValue = new IntPair (oldValue.lower, i); if (values.compareAndSet(oldValue,newValue)) { break ; } } } }
2.2 非阻塞的栈 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 public class ConcurrentStack <E> { private AtomicReference<Node<E>> top = new AtomicReference <>(); public void push (E item) { Node<E> newTop = new Node <>(item); Node<E> oldTop; do { oldTop = top.get(); newTop.next = oldTop; } while (!top.compareAndSet(oldTop, newTop)); } public E pop () { Node<E> oldTop; do { oldTop = top.get(); if (oldTop == null ) { return null ; } } while (!top.compareAndSet(oldTop, oldTop.next)); oldTop.next = null ; return oldTop.item; } private static class Node <E> { public final E item; public Node<E> next; public Node (E item) { this .item = item; } } }
2.3 非阻塞的队列 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 public class LinkQueue <E> { private final Node<E> dummy = new Node <>(null , null ); private final AtomicReference<Node<E>> head = new AtomicReference <>(dummy); private final AtomicReference<Node<E>> tail = new AtomicReference <>(dummy); public boolean put (E item) { Node<E> newNode = new Node <>(item, null ); while (true ) { Node<E> curTail = tail.get(); Node<E> next = curTail.next.get(); if (curTail != tail.get()) { continue ; } if (next != null ) { tail.compareAndSet(curTail, next); } else { if (curTail.next.compareAndSet(null , newNode)) { tail.compareAndSet(curTail, newNode); return true ; } } } } public E pop () { while (true ) { Node<E> curHead = head.get(); Node<E> firstNode = curHead.next.get(); if (firstNode == null ) { return null ; } Node<E> secondNode = firstNode.next.get(); if (curHead.next.compareAndSet(firstNode, secondNode)) { firstNode.next.set(null ); return firstNode.item; } } } private static class Node <E> { final E item; final AtomicReference<Node<E>> next; public Node (E item, Node<E> next) { this .item = item; this .next = new AtomicReference <>(next); } } }