mnlin
2018-03-02 02:42
采纳率: 20%
浏览 1.0k
已采纳

java原子性操作,volatile

在java中,并发考虑的是三个方面,原子性,可见性,有序性.
使用volatile可以保证可见性和有序性.
其中说到可见性时,提示说volatile修饰的变量在更改值的情况下,会将变量值直接同步到主内存中,并通知其他线程中该变量值失效,这一"直接同步到主内存"是如何实现的,是同时执行了assign -> store -> write三个操作才保证更改同步到主内存的么?可是虚拟机规范中规定的assign,store,write才是原子性操作,万一这三步执行中间时刻,有其他线程读取了数据,那怎么办?
还有一点,对于多线程来说:比如线程A和B,同时执行了对全局变量c的write原子操作,那怎么处理?谁会先执行?还是说在A线程对c变量执行store或者write操作时,其他线程无法对变量c进行任意的原子性操作?

  • 写回答
  • 关注问题
  • 收藏
  • 邀请回答

3条回答 默认 最新

  • oyljerry 2018-03-02 03:07
    已采纳

    所以说volatile没有原子性保证,如果有同时操作,就可能还是会读取到旧值,所以还是要靠synchronized来保证原子同步。
    但是如果是一个线程同步处理完了以后,另一个线程就会去读主内存,这样就能读取到新的值

    已采纳该答案
    打赏 评论
  • enigmatic_child 2018-03-02 03:52

    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);
    }
    

    }

    这里就会出错

    打赏 评论
  • 纪莫 2018-03-02 04:37

    在面试的时候很讨厌问这种问题的,虽然这么问,但是实际的工作中用到的还是很少

    打赏 评论

相关推荐 更多相似问题