在学习Volatile时遇到的一个问题,不知道如何解释

public class VolatileBoolean extends  Thread{

    private boolean b = true;

    @Override
    public void run() {
        while(b){
            System.out.println(b);
        }
        System.out.println("run 结束");
    }
    public static void main(String[] args) throws InterruptedException {
        VolatileBoolean volatileBoolean = new VolatileBoolean();
        volatileBoolean.start();
        Thread.sleep(1000);
        volatileBoolean.b = false;
        System.out.println("设置 false");
        Thread.sleep(1000);

    }
}

这个程序会正常结束 打印出 “run 结束”。
如果我们去掉System.out.println(b); 这句代码。

public class VolatileBoolean extends  Thread{

    private boolean b = true;

    @Override
    public void run() {
        while(b){
            //System.out.println(b);
        }
        System.out.println("run 结束");
    }
    public static void main(String[] args) throws InterruptedException {
        VolatileBoolean volatileBoolean = new VolatileBoolean();
        volatileBoolean.start();
        Thread.sleep(1000);
        volatileBoolean.b = false;
        System.out.println("设置 false");
        Thread.sleep(1000);

    }
}

代码执行结果却是不能停止 ,也不会打印 “run 结束”

自己分析不出为什么会这样,希望有人可以帮助理解一下

这个问题,前面也有一个相同问题,你可以看看:https://ask.csdn.net/questions/894382
当时我的回答是:

你可以将 isRunning 定义前面的 volatile 放开,那么该变量的可见性就能立即保证了,然后即使注释掉打印语句,也是能读到最新的变化值的。
楼主当前这种情况下,由于主线程 1 秒后操作了 isRunning 后,MyVolatile 线程并不能立即知道最新的值,而且 while 循环是空的,没有任何操作情况下,该线程访问的 isRunning 似乎一直都是自己用户空间的数据,而非共享空间的数据。
加上打印语句后,可能影响了变量的可见性使得其能够得到新得值。MyVolatile 的 while 循环中打印 isRunning 也能达到 volatile 的目的。
可以得出一个结论就是:无操作的空循环不能使得共享变量的值刷新成最新值。

这里将 b 定义为 volatile 就能保证可见性了。