在java中,并发考虑的是三个方面,原子性,可见性,有序性.
使用volatile可以保证可见性和有序性.
其中说到可见性时,提示说volatile修饰的变量在更改值的情况下,会将变量值直接同步到主内存中,并通知其他线程中该变量值失效,这一"直接同步到主内存"是如何实现的,是同时执行了assign -> store -> write三个操作才保证更改同步到主内存的么?可是虚拟机规范中规定的assign,store,write才是原子性操作,万一这三步执行中间时刻,有其他线程读取了数据,那怎么办?
还有一点,对于多线程来说:比如线程A和B,同时执行了对全局变量c的write原子操作,那怎么处理?谁会先执行?还是说在A线程对c变量执行store或者write操作时,其他线程无法对变量c进行任意的原子性操作?
所以说volatile没有原子性保证,如果有同时操作,就可能还是会读取到旧值,所以还是要靠synchronized来保证原子同步。
但是如果是一个线程同步处理完了以后,另一个线程就会去读主内存,这样就能读取到新的值
volatile没有原子性保证,只是保证了可见性和一定的有序性。
写入内存时,不会写入到CPU的L1或者L2缓存中对应的缓存,而是直接刷到主存中,
但是在多线程写入的时候,荏苒会发生错误,比如
public class Test {
public volatile int inc = 0;
public void increase() {
inc++;
}
public static void main(String[] args) {
final Test test = new Test();
for(int i=0;i<10;i++){
new Thread(){
public void run() {
for(int j=0;j<1000;j++)
test.increase();
};
}.start();
}
while(Thread.activeCount()>1) //保证前面的线程都执行完
Thread.yield();
System.out.println(test.inc);
}
}
这里就会出错
在面试的时候很讨厌问这种问题的,虽然这么问,但是实际的工作中用到的还是很少