《Java并发编程的艺术》 —3.6.5 为什么final引用不能从构造函数内“溢出”

举报
华章计算机 发表于 2019/12/03 16:28:21 2019/12/03
【摘要】 本节书摘来自华章计算机《Java并发编程的艺术》一书中第3章,第3.6.5节,作者是方腾飞 魏鹏 程晓明。

3.6.5 为什么final引用不能从构造函数内“溢出”

前面我们提到过,写final域的重排序规则可以确保:在引用变量为任意线程可见之前,该引用变量指向的对象的final域已经在构造函数中被正确初始化过了。其实,要得到这个效果,还需要一个保证:在构造函数内部,不能让这个被构造对象的引用为其他线程所见,也就是对象引用不能在构造函数中“逸出”。为了说明问题,让我们来看下面的示例代码。

public class FinalReferenceEscapeExample {

    final int i;

    static FinalReferenceEscapeExample obj;

 

    public FinalReferenceEscapeExample () {

        i = 1;                       // 1写final域

        obj = this;               // 2 this引用在此"逸出"

    }

 

    public static void writer() {

        new FinalReferenceEscapeExample ();

    }

 

    public static void reader() {

        if (obj != null) {              // 3

            int temp = obj.i;          // 4

        }

    }

}

 image.png

图3-31 引用型final的执行时序图

假设一个线程A执行writer()方法,另一个线程B执行reader()方法。这里的操作2使得对象还未完成构造前就为线程B可见。即使这里的操作2是构造函数的最后一步,且在程序中操作2排在操作1后面,执行read()方法的线程仍然可能无法看到final域被初始化后的值,因为这里的操作1和操作2之间可能被重排序。实际的执行时序可能如图3-32所示。

 image.png

图3-32 多线程执行时序图

从图3-32可以看出:在构造函数返回前,被构造对象的引用不能为其他线程所见,因为此时的final域可能还没有被初始化。在构造函数返回后,任意线程都将保证能看到final域正确初始化之后的值。


【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。