- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我尝试理解最终字段的语义。
让我们研究代码:
public class App {
final int[] data;
static App instance;
public App() {
this.data = new int[]{1, 0};
this.data[1] = 2;
}
public static void main(String[] args) {
new Thread(new Runnable() {
public void run() {
instance = new App();
}
}).start();
while (instance == null) {/*NOP*/}
System.out.println(Arrays.toString(instance.data));
}
}
我有一些问题:
P.S.我不知道如何使标题正确,请随意编辑。
如果我们替换:是否有能见度差异:
public App() {
this.data = new int[]{1, 0};
this.data[1] = 2;
}
与
public App() {
int [] data = new int[]{1, 0};
data[1] = 2;
this.data = data;
}
我还想知道如果在我的示例中将 final
替换为 volatile,wjat 会是什么。
因此我想得到关于 4 个新案例的解释
最佳答案
是的,有一些收获。您正在循环后重新读取 instance
变量,并且由于两次读取都是活泼的,退出循环并不能保证循环后的读取读取非 null
引用.
由于这个问题不是问题的主题,假设有以下变化:
App instance;
while((instance=App.instance) == null) {/*NOP*/}
System.out.println(Arrays.toString(instance.data));
然后,如果应用程序终止,它将输出[1,2]
。关键是 final
字段语义适用于整个构造函数,数组引用写入字段的确切时间无关紧要。这也意味着在构造函数中,重新排序是可能的,因此如果 this
引用在构造函数完成之前转义,则所有保证都是无效的,无论 this
是否在之前转义或者在程序顺序写入之后。由于在您的代码中,this
在构造函数完成之前不会转义,因此保证适用。
引用JLS §17.5., final
Field Semantics :
An object is considered to be completely initialized when its constructor finishes. A thread that can only see a reference to an object after that object has been completely initialized is guaranteed to see the correctly initialized values for that object's
final
fields.
请注意,它指的是完全初始化 状态,而不是写入特定的final
字段。这也将在下一节 §17.5.1 中解决。 :
Let o be an object, and c be a constructor for o in which a
final
field f is written. A freeze action onfinal
field f of o takes place when c exits, either normally or abruptly.
如果将变量更改为 volatile
,则几乎没有任何保证。 volatile
字段在对该变量的写入和后续读取之间建立了一种发生在之间的关系,但经常被忽视的关键点是“后续”。如果 App
实例发布不当,就像在您的示例中一样,则不能保证主线程对 instance.data
的读取将是后续的。如果它读取一个 null
引用,这现在是可能的,那么您就知道它不是后续的。如果它读取一个非 null
引用,您知道它是在字段写入之后的,这意味着您可以保证在第一个槽中读取 1
,但是第二次你可能会读到 0
或 2
。
如果你想从障碍和重新排序的角度讨论这个问题,volatile
写入 data
保证所有先前的写入都已提交,其中包括 1
到第一个数组槽,但它不保证后续的非 volatile
写入不会更早提交。因此,App
引用的不正确发布仍有可能在 volatile
写入之前执行(尽管这种情况很少发生)。
如果将写入移动到构造函数的末尾,一旦看到非null
数组引用,所有之前的写入都是可见的。对于 final
字段,不需要进一步讨论,如上所述,write 在构造函数中的实际位置无论如何都是无关紧要的。对于 volatile
情况,如上所述,您不能保证读取非 null
引用,但是当您读取它时,所有先前的写入都会被提交。了解表达式 new int[]{1, 0};
被编译为等价于 hiddenVariable=new int[2]; 可能会有所帮助。隐藏变量[0]=1; hiddenVariable[1]=0;
无论如何。在其构造之后但在对字段的数组引用的 volatile
写入之前放置另一个数组写入不会改变语义。
关于java - JMM 保证关于对象的最终引用和非最终引用,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41955348/
一 点睛 并发编程的三大特性是原子性、可见性和顺序性。 JVM 采用内存模型的机制来屏蔽各个平台和操作系统之间内存访问的差异,以实现让 Java 程序在各个平台下达到一致的内存访问效果,比如 C 语言
我多次看到这样的结构: class OneTimeWriter { volatile Object o; void synchronized set(Object o) {
这里 http://docs.oracle.com/javase/specs/jls/se8/html/jls-17.html#jls-17.4.8 是什么意思 hbi|Ci = hb|Ci soi|
我试图从 Java 内存模型的角度理解“顺序一致性”。 JLS 的定义,第 17 章对我来说不是那么清楚。 我会给出我的看法,如果我错了请纠正我。具有单线程顺序一致性的程序意味着如果 action1
我有一个关于 JMM 的问题。我知道什么是“之前发生”,但我不明白一件事。有一段代码: private static int x = 0; private static volatile boolea
考虑以下用 Java 编写的多线程代码: 共享变量: boolean n; // non-volatile volatile boolean v; // volatile 线
我正在尝试了解 JMM 的因果关系要求。 JMM 的这一部分描述了它 https://docs.oracle.com/javase/specs/jls/se7/html/jls-17.html#jls
这里 [ http://docs.oracle.com/javase/specs/jls/se7/html/jls-17.html#jls-17.5][1] 它说: Given a write w,
我尝试理解最终字段的语义。 让我们研究代码: public class App { final int[] data; static App instance; public
我最近偶然发现了 this jcstress 中的示例: @JCStressTest @State @Outcome(id = "10", expect =
这篇文章是在阅读后提出的:https://shipilev.net/blog/2016/close-encounters-of-jmm-kind/#pitfall-semi-sync class Bo
我正在寻找一种将对象从后台线程传递到 UI 线程的安全方法。下面的代码是否安全? // on background thread final HugeObject object = construct
我正在阅读有关 java singleton 的内容,并且遇到了一些奇怪的事情。 我会引用following artice例如(您可以轻松找到更多) 作者提供了以下单例: public class A
以下是 volatile 关键字的使用示例。 public class Test3{ public static volatile boolean stop = false;// if vol
请考虑以下代码片段。 public class Visibility { private volatile SomeData data; public static class Som
对 trylock 的不当使用 T1 T2 x = 42; while (
我只是想知道下面的重新排序在新的 JMM 模型下是否有效 Original Code: instanceVar1 = value ;// normal read operation, no
关闭。这个问题需要更多 focused .它目前不接受答案。 想改进这个问题?更新问题,使其仅关注一个问题 editing this post . 2年前关闭。 Improve this questi
假设这个类: public class AmIThreadSafe { private int a; private int b; AmIThreadSafe(int a, i
JLS 中写到关于 JMM 的 happens-before 关系(第 17.4.5 节): It should be noted that the presence of a happens-bef
我是一名优秀的程序员,十分优秀!