1、理解原子性:
上面说到volatile不具备原子性,那么原子性到底是什么呢?先看如下代码
public class TestVolatile {public static void main(String[] args) {AtomicDemo atomicDemo = new AtomicDemo();for (int x = 0; x < 10; x++) {new Thread(atomicDemo).start();}}
}class AtomicDemo implements Runnable {private int i = 0;public int getI() {return i++;}@Overridepublic void run() {try {Thread.sleep(200);} catch (InterruptedException e) {e.printStackTrace();}System.out.println(getI());}
}
这段代码就是在run方法里面让i++,然后启动十个线程去访问。看看结果:
可以发现,出现了重复数据。明显产生了多线程安全问题,或者说原子性问题。所谓原子性就是操作不可再细分,而i++操作分为读改写三步,如下:
int temp = i;
i = i+1;
i = temp;
所以i++明显不是原子操作。上面10个线程进行i++时,内存图解如下:
看到这里,好像和上面的内存可见性问题一样。是不是加个volatile关键字就可以了呢?其实不是的,因为加了volatile,只是相当于所有线程都是在主存中操作数据而已,但是不具备互斥性。比如两个线程同时读取主存中的0,然后又同时自增,同时写入主存,结果还是会出现重复数据