大家好,我是鸭鸭!
此答案节选自鸭鸭最近弄的面试鸭神器面试鸭,更多大厂常问面试题,可以点击进行阅读哈!
目前这个面试刷题小程序刚出!
回归面试题:final 可以保证可见性吗?
不可以。
你可能看到一些答案说可以保证可见性,那不是我们常说的可见性。
一般而言我们指的可见性是一个线程修改了共享变量,另一个线程可以立马得知更改,得到最新修改后的值。
而 final 并不能保证这种情况的发生,volatile 才可以。
而有些答案提到的 final 可以保证可见性,其实指的是 final 修饰的字段在构造方法初始化完成,并且期间没有把 this 传递出去,那么当构造器执行完毕之后,其他线程就能看见 final 字段的值。
如果不用 final 修饰的话,那么有可能在构造函数里面对字段的写操作被排序到外部,这样别的线程就拿不到写操作之后的值。
来看个代码就比较清晰了。
public class YesFinalTest {
final int a;
int b;
static YesFinalTest testObj;
public void YesFinalTest () { //对字段赋值
a = 1;
b = 2;
}
public static void newTestObj () { // 此时线程 A 调用这个方法
testObj = new YesFinalTest ();
}
public static void getTestObj () { // 此时线程 B 执行这个方法
YesFinalTest object = obj;
int a = object.a; //这里读到的肯定是 1
int b = object.b; //这里读到的可能是 2
}
}
对于 final 域,编译器和处理器要遵守两个重排序规则(参考自infoq程晓明):
- 在构造函数内对一个 final 域的写入,与随后把这个被构造对象的引用赋值给一个引用变量,这两个操作之间不能重排序。 初次读一个包含
- final 域的对象的引用,与随后初次读这个 final 域,这两个操作之间不能重排序。
所以这才是 final 的可见性,这种可见性和我们在并发中常说的可见性不是一个概念!
所以 final 无法保证可见性!
最后
最后再推荐下鸭鸭目前努力在做面试神器面试鸭,已经有 3000 多道面试题目啦,欢迎大家来阅读!如果大家有不会的面试题,也可以在面试鸭内反馈!鸭鸭会第一时间为大家解答!
我是鸭鸭,我们下期见~
946

被折叠的 条评论
为什么被折叠?



