赫尔本 2019-10-30 20:37 采纳率: 0%
浏览 435
已结题

volatile/ java内存模型/ JMM ---代码求解

在学习volatile变量时碰到的疑惑:


1、如下图,按照jmm的理论,因为 changed 这个变量未被volatile修饰,所以子程序的while直接死循环

图片说明

public class ThreadTest {
    // 注意,未被volatile 修饰
    private static boolean changed = false;

    public static void main(String[] args) throws InterruptedException {
        new Thread(() -> {
            System.out.println("子线程变量值: changed = " + changed);
            while (!changed) {
                // 因为changed值未被volatile修饰,所以读到的changed一直是本地线程变量值
            }
            System.out.println("子线程结束。。。");
        }).start();

        System.out.println("主线程变量值:changed = " + changed);

        // sleep 100毫秒,确保子线程已经进入了19行的死循环
        Thread.sleep(100);
        changed = true;
        System.out.println("主线程结束。。。changed = " + changed);

    }
}

2、但是如果在while循环里加上sleep,程序能正常结束

图片说明


public class ThreadTest {
    // 注意,未被volatile 修饰
    private static boolean changed = false;

    public static void main(String[] args) throws InterruptedException {
        new Thread(() -> {
            System.out.println("子线程变量值: changed = " + changed);
            while (!changed) {
                // 因为changed值未被volatile修饰,所以读到的changed一直是本地线程变量值
                try {
                    Thread.sleep(50);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }

            }
            System.out.println("子线程结束。。。changed = " + changed);
        }).start();

        System.out.println("主线程变量值:changed = " + changed);

        // sleep 100毫秒,确保子线程已经进入了19行的死循环
        Thread.sleep(100);
        changed = true;
        System.out.println("主线程结束。。。changed = " + changed);

    }
}


3、所以这是为什么,sleep操作会做触发线程的变量的reload行为?


4、官方文档

图片说明


5、本人vx:17610360027,求大佬解答,困惑多日

  • 写回答

2条回答 默认 最新

  • 关注
    评论
  • 深巷里的黑猫 2019-10-31 08:47
    关注

    这个问题我上次正好遇见过。
    由于while循环执行空语句,因此导致change访问频率过高,change不能及时的被写入主存中。这就是volatile可见性的一个原因,如果使用volatile则修改的值会立即被更新主存中。而增加一条语句之后,访问change具有一定的间隔,主内存就会有时间刷新工作内存中的change的值。 这也是JMM的工作原理,可以看看有关的文章来了解一下。

    看第一个人引用的文章链接里面,有一段很重要的话

    由于JVM运行程序的实体是线程,而每个线程创建时JVM都会为其创建一个工作内存(有些地方称为栈空间),用于存储线程私有的数据,而Java内存模型中规定所有变量都存储在主内存,主内存是共享内存区域,所有线程都可以访问,但线程对变量的操作(读取赋值等)必须在工作内存中进行,首先**要将变量从主内存拷贝的自己的工作内存空间,然后对变量进行操作,操作完成后再将变量写回主内存,不能直接操作主内存中的变量**,工作内存中存储着主内存中的变量副本拷贝,前面说过,**工作内存是每个线程的私有数据区域,因此不同的线程间无法访问对方的工作内存,线程间的通信(传值)必须通过主内存来完成**

    并且后面也有提到过

    需要注意的是,在主内存中的实例对象可以被多线程共享,倘若两个线程同时调用了同一个对象的同一个方法,那么两条线程会将要操作的数据拷贝一份到自己的工作内存中,执行完成操作后才刷新到主内存

    这几句话就解决了你的问题。 实际上你在纠结的是sleep吧??? 实际上你不需要写sleep也行,你随便写一条打印语句,他也能够正常结束

    评论

报告相同问题?

悬赏问题

  • ¥15 win11系统打开软件很慢
  • ¥30 XIAO esp32c3 读取FDC2214的数据
  • ¥20 我用malloc申请了一块空间 判空显示指针不为null 但是在输出data指针所指的地址是缺全是0 空指针不能输入值进去数组为什么呀
  • ¥15 在工控机(Ubuntu系统)上外接USB蓝牙硬件进行蓝牙通信
  • ¥15 关于PROCEDURE和FUNCTION的问题
  • ¥100 webapi的部署(标签-服务器)
  • ¥20 怎么加快手机软件内部计时的时间(关键词-日期时间)
  • ¥15 C语言除0问题的检测方法
  • ¥15 为什么四分管的内径有的是16mm有的15mm,四分不应该是12.7mm吗
  • ¥15 macos13下 ios交叉编译的问题