为什么FutureTask类的outcome变量不需要加上volatile修饰?

发布于 2022-09-11 20:17:35 字数 679 浏览 38 评论 0

第一个问题:
为什么outcome变量不加上volatile,以及在set和setException方法中,要专门的UNSAFE.putOrderedInt 已经加上volatile的state方法?

因为我个人认为,outcomme会被多个线程访问(一个可以读写,其他的只能读),这种情况下,为啥不加上volatile,加上volatile的好处可以让outcome和state变量被修改后,其他线程立刻知道?而为什么作者故意不使用volatile的特性呢?

    /** The result to return or exception to throw from get() */
    private Object outcome; // non-volatile, protected by state reads/writes
        
    protected void set(V v) {
        if (UNSAFE.compareAndSwapInt(this, stateOffset, NEW, COMPLETING)) {
            outcome = v;
            UNSAFE.putOrderedInt(this, stateOffset, NORMAL); // final state
            finishCompletion();
        }
    }

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(3

山有枢 2022-09-18 20:17:35

猜测:

state已经是volatile,写outcome happen-before 写state,根据内存模型别的线程看到state=NORMAL时应该也看到最新的outcome

只有影子陪我不离不弃 2022-09-18 20:17:35

我理解它内部已经有state来标识整个线程的运行状况,写入和读取都是受state的保护,换句话说,当我们读取完了需要返回结果的时候是先CASstate值,修改成NEW->COMPLETING,然后再去存放结果。

⊕婉儿 2022-09-18 20:17:35

在执行完outcome = v; 之后有一句
UNSAFE.putOrderedInt(this, stateOffset, NORMAL);
这个是把状态刷到主内存保证其他线程都可见。happens-before 原则outcome对其他线程也可见
详细可参考下:https://stackoverflow.com/questions/14432400/why-outcome-object-in-futuretask-is-non-volatile

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文