- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
根据 Java 内存模型,一个 final
在对象的构造函数中初始化的字段不受进一步修改的影响,保证每个读取它的线程都能正确看到它的值,即使对象本身已经发布了数据竞争。
JLS 谈论17.5.3 Subsequent Modification of Final Fields , 并含糊地说
An implementation may provide a way to execute a block of code in a final field safe context.
它似乎并没有真正定义这种修改的语义,也没有确切地定义这个 final 字段安全上下文 东西必须存在的地方或如何定义它(即,JLS 似乎没有给出对最终字段的后续修改的任何保证)。
我必须说,我没有完全理解部分命令 dereferences() 和 mc(),也没有完全理解 freeze 的行为在对最终字段进行任何修改(归因于它的初始值或后续修改)之后发生的操作。
在这种情况下,我想知道的是:(反)序列化框架(例如 Gson)如何保证包含在构造函数中正确初始化的最终字段的反序列化对象不会造成线程可见性问题?
例如,考虑这个类:
class X {
private final String s;
public X(final String s) { this.s = s; }
@Override public String toString() { return s; }
}
以及以下代码:
final Gson gson = new Gson();
X x = gson.fromJson(gson.toJson(new X("abc")), X.class);
System.out.println(x);
// prints abc
进入方法fromJson
使用调试器,我看到 sun.misc.Unsafe
用于分配 X
的实例无需调用其构造函数,字段为 setAccessible(true)
, 最后他们设置好了。
而且这仅在 Sun 的(或兼容的)JVM 中!看起来 Gson 也有针对多个 Android 版本的代码。
那么,是否有与这些反序列化的最终字段相关的任何线程安全保证,就像我对 X
的实例所做的那样用 new X("abc")
构建?如果是,此保证从何而来?
谢谢!
最佳答案
线程安全
正如我所读,线程安全保证来自于给定属性被声明为最终属性这一事实。它不是线程安全的点是:
final
属性赋值之前final
字段(即,当值处于修改过程中,并且在此过程完成之前)这里需要注意的是,您链接到的引用允许存在一些其他而不是反射 API(但具有相同的final
字段修改能力)的理论可能性.
卡住
Freezes of a final field occur both at the end of the constructor in which the final field is set, and immediately after each modification of a final field via reflection or other special mechanism.
就卡住调用而言,基本上是说 freeze
用于将属性标记为“无法更改”,它确实这样做了:
final
字段实际被赋值final
字段值更改后,通过 Reflection API 之类的东西线程安全问题仅适用于被修改/反序列化的对象。
所以:
(...code that comes before...)
END final field safe context - entering non-threadsafe area
final fields are not frozen
code that deserializes an object OR modifies a final field via reflection
final fields are re-frozen
RESUME final field safe context - re-entering threadsafe area
(...code that comes after...)
因此……
在构造函数运行后(即你有一个分配了值的实例化对象),final 字段不能被改变,因为它被卡住,除非使用反射 API 直接改变该值 - 之后它再次被卡住,因为毕竟该字段被声明为最终字段。
如果您正在寻找特定于 Android 的答案(以确保其 JVM 行为与 Sun/Oracle 的 JVM 一致),您需要找到该 JVM 的等效文档。
关于java - 使用 setAccessible(true) 的最终字段语义和反序列化,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7654747/
我正在尝试提升我的 javascript 编程技能(或者说我的编程技能时期 :)) 所以我试图理解一些语义: 第一行的“?”是什么意思?均值和“-distance”中的减号 第二行中的“+=”或“-=
我正在尝试在语义 UI 中执行复选框,但它不起作用,我无法弄清楚我做错了什么。 我包括jquery、semantic.min.js、checkbox.js 和semantic.min.css,然后我添
我正在构建一个 Spring 后端。我有一个 Controller ,它获取一个“搜索对象” - 一个具有 10 个字段的对象,其中只有一个应该被填充,所以搜索功能(我没有编写,但需要对其进行更改和重
我面临着编写更智能/高级的“相关内容”算法的挑战,并且不知道从哪里开始,所以我决定提出一个问题,是否有人会指出我正确的方向。 我们的数据库包含很多文章,到目前为止,我们使用关键字/标签查询了相关文章,
我正在尝试将通用字符串写入Rust中的数字函数,其中支持的类型为i16,i32,i64,u32,u64,f32和f64。 最初我有这个: fn str_to_num(s: &str, default_
假设我们在 hpp 文件中有一个带有唯一指针的简单结构: struct SomeType { SomeType() = default; ~SomeType(); st
这是同一预处理指令的多个问题。 1 - <> 还是 ""? 除了在 MSDN 中找到的信息: #include Directive (C-C++) 1.a:这两种符号有什么区别? 1.b:所有编译器都
所以基本上我有一个带有列表的简单系统,当我选择一个项目时,它会显示描述和绑定(bind)到该项目的图像。 项目:https://jsfiddle.net/jhnjcddh/2/ 问题是我需要在 JS
很抱歉问了一个愚蠢的问题,但有人能告诉我以下是什么意思吗 for ctype, (codename, name) in searched_perms: 我不明白括号里是怎么回事。 for ctype
这个问题在这里已经有了答案: 关闭 11 年前。 Possible Duplicate: Why do all these crazy function pointer definitions al
我正在学习 HTML5,并获得了一个将 CSS Zen Gardens 转换为 HTML5 语义版本的项目。我已经能够轻松地转换其中的大部分内容,但是底部的链接/导航给我带来了一些问题。 转换此/处理
关闭。这个问题是opinion-based .它目前不接受答案。 想要改进这个问题? 更新问题,以便 editing this post 可以用事实和引用来回答它. 关闭 9 年前。 Improv
我一直在我的 emacs c/c++ 开发设置中试验 cedet 和语义,除了一个小细节外,我对它非常满意。 我使用 ede-cpp-root-project 创建一个项目,并给出我的项目的根目录以及
引用问题自http://www.garfieldtech.com/blog/put-up-with-put (这是针对 Drupal 开源项目的,有点元,因为这里没有代码): GET、HEAD 和 P
我有以下代码。 let v_blue = UIView() v_blue.backgroundColor = UIColor.blueColor() l
我目前正在 objc.io 上阅读优秀的 Advanced Swift 书籍,但遇到了一些我不明白的问题。 如果您在操场上运行以下代码,您会注意到在修改字典中包含的结构时,下标访问会生成一个副本,但随
谁能给我一个关于 Flutter 上下文中语义概念的清晰解释(或链接)(它实际上是什么,何时使用,更新...)? 我在谷歌上搜索了很多,但到目前为止还没有找到任何好的解释。 非常感谢, 最佳答案 Di
这是我的代码 Was this what you wanted? It's good to see you again.
我有一个侧边栏,其中包含应用程序的主导航。它还包含一个 button 触发侧边栏的打开/关闭。在语义方面,标记应该是什么样的? 我应该把侧边栏放在一边,然后只在周围设置导航吗主导航,不包括打开/关闭触
考虑下面这行 Lisp 代码: (some-function 7 8 | 9) ;; some comment. note the extra indentation 该点位于“8”和
我是一名优秀的程序员,十分优秀!