1、何谓Atomic?java
Atomic一词跟原子有点关系,后者曾被人认为是最小物质的单位。计算机中的Atomic是指不能分割成若干部分的意思。若是一段代码被认为是Atomic,则表示这段代码在执行过程当中,是不能被中断的。一般来讲,原子指令由硬件提供,供软件来实现原子方法(某个线程进入该方法后,就不会被中断,直到其执行完成)安全
在x86 平台上,CPU提供了在指令执行期间对总线加锁的手段。CPU芯片上有一条引线#HLOCK pin,若是汇编语言的程序中在一条指令前面加上前缀"LOCK",通过汇编之后的机器代码就使CPU在执行这条指令的时候把#HLOCK pin的电位拉低,持续到这条指令结束时放开,从而把总线锁住,这样同一总线上别的CPU就暂时不能经过总线访问内存了,保证了这条指令在多处理器环境中的原子性。多线程
2、JDK1.5的原子包:java.util.concurrent.atomic并发
这个包里面提供了一组原子类。其基本的特性就是在多线程环境下,当有多个线程同时执行这些类的实例包含的方法时,具备排他性,即当某个线程进入方法,执行其中的指令时,不会被其余线程打断,而别的线程就像自旋锁同样,一直等到该方法执行完成,才由JVM从等待队列中选择一个另外一个线程进入,这只是一种逻辑上的理解。其实是借助硬件的相关指令来实现的,不会阻塞线程(或者说只是在硬件级别上阻塞了)。其中的类能够分红4组dom
Atomic类的做用ide
2.1 AtomicBoolean , AtomicInteger, AtomicLong, AtomicReference函数
这四种基本类型用来处理布尔,整数,长整数,对象四种数据。this
2.1.1 1个例子-使用AtomicReference建立线程安全的堆栈atom
public class LinkedStack<T> { private AtomicReference<Node<T>> stacks = new AtomicReference<Node<T>>(); public T push(T e) { Node<T> oldNode, newNode; while (true) { //这里的处理很是的特别,也是必须如此的。 oldNode = stacks.get(); newNode = new Node<T>(e, oldNode); if (stacks.compareAndSet(oldNode, newNode)) { return e; } } } public T pop() { Node<T> oldNode, newNode; while (true) { oldNode = stacks.get(); newNode = oldNode.next; if (stacks.compareAndSet(oldNode, newNode)) { return oldNode.object; } } } private static final class Node<T> { private T object; private Node<T> next; private Node(T object, Node<T> next) { this.object = object; this.next = next; } } }
2.1.2 几个问题spa
Q1: compareAndSet和weakCompareAndSet的区别?
A1: 有人认为这是个坑,由于这2个方法其中的内容是如出一辙的。疑惑ing(环境JDK1.6.0_20_b02)
Q2:volatile boolean和AtomicBoolean的区别?
Q3:volatile int和AtomicInteger的区别?
Q4:LazySet()和Set()的区别?
3、Atomic举例
3.1 原子量实现的计数器
import java.util.HashSet; import java.util.Set; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.atomic.AtomicInteger; public class AtomicCounter { private AtomicInteger value = new AtomicInteger(); public int getValue() { return value.get(); } public int increase() { return value.incrementAndGet();// 内部使用死循环for(;;)调用compareAndSet(current, next) // return value.getAndIncrement(); } public int increase(int i) { return value.addAndGet(i);// 内部使用死循环for(;;)调用compareAndSet(current, next) // return value.getAndAdd(i); } public int decrease() { return value.decrementAndGet();// 内部使用死循环for(;;)调用compareAndSet(current, next) // return value.getAndDecrement(); } public int decrease(int i) { return value.addAndGet(-i);// 内部使用死循环for(;;)调用compareAndSet(current, next) // return value.addAndGet(-i); } public static void main(String[] args) { final AtomicCounter counter = new AtomicCounter(); ExecutorService service = Executors.newCachedThreadPool(); for (int i = 0; i < 10; i++) { service.execute(new Runnable() { @Override public void run() { System.out.println(counter.increase()); } }); } service.shutdown(); } }
3.2 原子量实现的银行取款
import java.util.Random; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.atomic.AtomicLong; public class Account { private AtomicLong balance; public Account(long money) { balance = new AtomicLong(money); System.out.println("Total Money:" + balance); } public void deposit(long money) { balance.addAndGet(money); } public void withdraw(long money) { for (; ; ) {//保证即时同一时间有人也在取款也能够再次尝试取款,若是不须要并发尝试取款,能够去掉这句 long oldValue = balance.get(); if (oldValue < money) { System.out.println(Thread.currentThread().getName() + " 余额不足! 余额:" + balance); break; } try {Thread.sleep(new Random().nextInt(1000));} catch (Exception e) { }// 模拟取款时间 if (balance.compareAndSet(oldValue, oldValue - money)) { System.out.println(Thread.currentThread().getName() + " 取款 " + money + " 成功! 余额:" + balance); break; } System.out.println(Thread.currentThread().getName() + " 遇到并发,再次尝试取款!"); } } public static void main(String[] args) { final Account account = new Account(1000); ExecutorService pool = Executors.newCachedThreadPool(); int i = 0; while (i++ < 13) { pool.execute(new Runnable() { @Override public void run() { account.withdraw(100); } }); } pool.shutdown(); } }