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

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
    关注

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

    评论
查看更多回答(2条)

报告相同问题?

悬赏问题

  • ¥15 请教如何为VS2022搭建 Debug|win32的openCV环境?
  • ¥15 关于#c++#的问题:c++如何使用websocketpp实现websocket接口调用,求示例代码和相关资料
  • ¥15 51单片机的外部中断,按下按键后不能切换到另一个模式
  • ¥15 java连接sqlserver有问题
  • ¥15 yolov8 如何调cfg参数
  • ¥15 这个四人抢答器代码哪儿有问题?仿真程序怎么写?
  • ¥15 burpsuite密码爆破
  • ¥15 关于#ubuntu#的问题,如何解决?(相关搜索:移动硬盘)
  • ¥15 scikit安装之后import不了
  • ¥15 Ros2编译一个使用opencv的c++节点的时候,报了这个错误,请问怎么解决啊