关于 volatile 可见性的一个问题

2020-10-05 11:19:04 +08:00
 mtmax

为啥线程读取了一个 volatile 变量 b, 居然能同时读到非 volatile 变量 a 的最新值

static long a = 0;

static long p1, p2, p3, p4, p5, p6, p7, p8, p9, p10, p11, p12, p13, p14, p15, p16;

static volatile long b = 0;

public static void main(String[] args) throws InterruptedException {

    new Thread(() -> {
        while (a == 0) {
            long x = b; // 为什么这里读 b, 能让线程同时读到 a 的最新值?  如果注释这行, a 就读不到
        }
        System.out.println("a=" + a);
    }).start();

    Thread.sleep(100);

    a = 1;
}
3326 次点击
所在节点    Java
25 条回复
Weixiao0725
2020-10-06 07:37:50 +08:00
因为 a 只是普通变量,什么时候刷新到内存要看运行时。当你加上读取 b 的那句代码时,因为 b 是 volatile 的,会强制从内存读,所以这时候强制把 a 的内容重新刷到了内存中,所以这时候就可以读取到最新的 a 值了
matt5ttam
2020-10-06 11:10:03 +08:00
这个是缓存一致性协议造成的 volatile 会使用 lock#锁总线
Newyorkcity
2020-10-06 15:22:11 +08:00
我本来想说会不会是缓存行(字)的问题..

但楼主的代码里之所以会有

static long p1, p2, p3, p4, p5, p6, p7, p8, p9, p10, p11, p12, p13, p14, p15, p16;

想必也是考虑到这个方向了吧....而且看这个样子应该也不是缓存行的问题了.
letianqiu
2020-10-06 16:34:24 +08:00
letianqiu
2020-10-06 16:39:14 +08:00
@Wicked 这么多楼就你一个明白人。

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/712592

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX