- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
这是在 java 多线程中安全构造实践的上下文中。我正在阅读 JCIP 书,有人可以在下面解释一下吗:
an object is in a predictable, consistent state only after its constructor returns, so publishing an object from within its constructor can publish an incompletely constructed object. This is true even if the publication is the last statement in the constructor.
在这里我想了解具体的部分'即使发布是这样也是如此 构造函数中的最后一条语句'
我知道从构造函数启动一个线程是不安全的,但是当它是构造函数中的最后一条语句时它也不安全吗?
我的猜测是关于为什么它不安全的原因是,JVM 可以对语句进行重新排序,而最后一个语句将不再是最后一个。请对此发表评论或纠正我的理解。
编辑:有问题的是,启动线程的意思是泄漏 this
引用。这种泄漏可能通过多种方式发生,例如发布事件或启动线程。
最佳答案
这里有两个见解可能对您有所帮助。首先,出于同步目的,构造函数(大部分)与任何其他方法一样;也就是说,它本身并不提供任何(除非如下所述)。其次,线程安全始终存在于各个操作之间。
假设您有以下构造函数:
MyClass() {
this.i = 123;
MyClass.someStaticInstance = this; // publish the object
}
// and then somewhere else:
int value = MyClass.someStaticInstance.i;
问题是:最后一个表达式能做什么?
someStaticInstance
尚未设置。value == 123
value == 0
最后一点的原因是 Action 可以重新排序,构造函数在这方面并不特殊。让我们仔细看看所涉及的操作:
int i
)<instance>.i = 123
someStaticInstance = <instance>
someStaticInstance
,然后是它的 i
如果你稍微重新排序,你可以得到:
int i
)someStaticInstance = <instance>
someStaticInstance
,然后是它的 i
<instance>.i = 123
这就是你的 -- value
是 0,而不是 123。
JCIP 还警告您泄漏可能以微妙的方式发生。例如,假设您没有明确设置 someStaticInstance
字段,而只是调用 someListener.register(this)
.您仍然泄露了引用,并且您应该假设您注册的监听器可能会对它做一些危险的事情,比如将它分配给 someStaticInstance
.
即使i
也是如此字段是最终的。您可以从 final 字段获得一些线程安全性,但前提是您不泄漏 this
来自构造函数。具体来说,在 JLS 17.5 ,它说:
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.
保证“只有在对象完全初始化后才能看到对该对象的引用”的唯一方法是不从其构造函数中泄漏引用。否则,您可以想象一个线程读取 MyClass.someStaticInstance
在设置字段之后,但在 JVM 将构造函数识别为已完成之前。
关于java - java多线程中的安全构造,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48171140/
谁能解释一下原因: (define a (lambda() (cons a #f))) (car (a)) ==> procedure ((car (a))) ==> (procedure . #f)
这是 PyBrain 网站的摘录。我了解大部分正在发生的事情,但是一行让我完全难住了。我以前从未在 python 代码中看到过这样的东西。这是整个循环,对于上下文: for c in [0,
我是gradle / groovy的新手。我想创建将做一些事情的自定义任务。我的第一个问题是任务完成时该如何做?我可以覆盖doFirst / doLast闭包吗?也许我可以重写某些在开始和结束时都会执
我刚刚开始评估 MS 企业库。他们使用以下指令来获取实例: var customerDb = EnterpriseLibraryContainer.Current.GetInstance("C
这是我的 if else Ansible 逻辑.. - name: Check certs exist stat: path=/etc/letsencrypt/live/{{ rootDomain
我正在使用construct 2.8 对一些失传已久的 Pascal 程序创建的一些文件的 header 进行逆向工程。 header 由许多不同的记录组成,其中一些是可选的,我不确定顺序是否固定。
我在将 getchar() 的输入放入 char *arr[] 数组时遇到问题。我这样做的原因是因为输入数据(将是一个带有命令行参数的文件)将存储在一个 char 指针数组中以传递给 execvp 函
通常我们不能约束类型参数 T派生自密封类型(例如 struct 类型)。这将毫无意义,因为只有一种类型适合,因此不需要泛型。所以约束如下: where T : string 或: where T :
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 9 年前。 Improve th
#include using namespace std; class A { private: int m_i; friend int main(int argc, char cons
这个问题在这里已经有了答案: Are there legitimate uses for JavaScript's "with" statement? (33 个答案) 关闭 9 年前。 我有这个代
在this answer我看到了下一个 Bash 结构。 yes "$(< file.txt)" 什么意思 "$(< file.txt)" ? 我明白了 命令替换 - $(command)用命令的结
if (a == 1) //do something else if (a == 2) //do something else if (a == 3) //do somethi
关于构造的快速简单的问题。 我有以下用于将项目添加到 ListView 的代码。 ListViewItem item = new ListViewItem(); item.Text = file; i
我想使用 std::vector 来控制给定的内存。首先,我很确定这不是好的做法,但好奇心占了上风,无论如何我都想知道如何做到这一点。 我遇到的问题是这样的方法: vector getRow(unsi
下面显示了一段简单的javascript: var mystring = ("random","ignored","text","h") + ("ello world") 这个字符串会生成 hello
在 Java 中,创建对象的标准方法是使用 MyClass name = new MyClass(); 我也经常看到构造 new MyClass() { /*stuff goes in here*/
我正在编写 C++ ndarray 类。我需要动态大小和编译时大小已知的数组(分别分配自由存储和分配堆栈)。我想支持从嵌套的 std::initializer_list 进行初始化。 动态大小的没问题
我正在将一个项目从 Visual Studio 2005 转换为 Visual Studio 2008,并提出了上述结构。 using Castle.Core.Resource; using Cast
我想知道我在这里的想法是否正确,我主要针对接口(interface)进行编程,所以我想知道下面的类是否应该通过 DI 注入(inject),或者我应该自己实例化一个类... 注意:这些服务保存在我的核
我是一名优秀的程序员,十分优秀!