- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我已经阅读了很多有关Java同步以及可能发生的所有问题的文章。但是,我仍然有些困惑的是JIT如何重新排序写入。
例如,一个简单的双重检查锁对我来说很有意义:
class Foo {
private volatile Helper helper = null; // 1
public Helper getHelper() { // 2
if (helper == null) { // 3
synchronized(this) { // 4
if (helper == null) // 5
helper = new Helper(); // 6
}
}
return helper;
}
}
helper
,但是,构造函数尚未运行,因为JIT可以对我们的代码进行重新排序。 MyObject
并将其放入
HashMap<String, MyObject>
的方法(我知道
HashMap
并非线程安全的,不应在多线程环境中使用,但请耐心等待)。线程1调用createNewObject:
public class MyObject {
private Double value = null;
public MyObject(Double value) {
this.value = value;
}
}
Map<String, MyObject> map = new HashMap<String, MyObject>();
public void createNewObject(String key, Double val){
map.put(key, new MyObject( val ));
}
public MyObject getObject(String key){
return map.get(key);
}
getObject(String key)
接收未完全构造的对象?就像是:
new MyObject( val )
分配内存getObject(String key)
map.put(key, new MyObject( val ))
不会在完全构建对象之前将其放入 map 中?
Object
并将其分配给引用变量(例如经过双重检查的锁)时重新排序吗?一个完整的JIT概要对于一个SO答案来说可能很多,但是我真正好奇的是它如何重新排序一个写操作(如双重检查锁的第6行),以及阻止它将对象放入
Map
的原因。尚未完全建成。
最佳答案
警告:文字墙
您的问题的答案在水平线之前。我将在答案的第二部分(与JIT无关,因此仅在您对JIT感兴趣的情况下就这样)中更深入地解释基本问题。问题第二部分的答案位于底部,因为它取决于我进一步描述的内容。
TL; DR在您通过编写线程不安全代码让它们有效的条件下,JIT可以执行任何所需的操作,JMM可以执行所需的任何操作。
注意:“初始化”是指构造函数中发生的事情,它不包括其他任何事情,例如在构造之后调用静态init方法等。
"If the reordering produces results consistent with a legal execution, it is not illegal." (JLS 17.4.5-200)
"The memory model describes possible behaviors of a program. An implementation is free to produce any code it likes, as long as all resulting executions of a program produce a result that can be predicted by the memory model.
This provides a great deal of freedom for the implementor to perform a myriad of code transformations, including the reordering of actions and removal of unnecessary synchronization" (JLS 17.4).
"If Helper is an immutable object, such that all of the fields of Helper are final, then double-checked locking will work without having to use volatile fields. The idea is that a reference to an immutable object (such as a String or an Integer) should behave in much the same way as an int or float; reading and writing references to immutable objects are atomic" (The "Double-Checked Locking is Broken" Declaration).
"There are a few trivial ways to achieve safe publication:
- Exchange the reference through a properly locked field (JLS 17.4.5)
- Use static initializer to do the initializing stores (JLS 12.4)
- Exchange the reference via a volatile field (JLS 17.4.5), or as the consequence of this rule, via the AtomicX classes
- Initialize the value into a final field (JLS 17.5)."
volatile
的原因是为了使线程1中对助手的写入相对于线程2中的读取是有序的。允许在读取后感知初始化,因为它发生在向辅助程序的写入之前。它背负着 volatile 写操作,以便必须在初始化之后进行读取,然后再对 volatile 字段(传递属性)进行写操作。
I'd imagine that the answer is, it wouldn't put an object into the Map until it is fully constructed (because that sounds awful). So how can the JIT reorder?
value
或将其添加到 map 之间没有顺序,因此线程2
会感知到的困惑,因为
MyObject
是不安全发布的。
ConcurrentHashMap
来解决此问题,并且
getObject()
将完全是线程安全的,因为一旦将对象放入 map 中,初始化将在put之前进行,并且由于
get
被执行,因此两者都需要在
ConcurrentHashMap
之前进行线程安全。但是,一旦修改了对象,这将成为管理方面的噩梦,因为您需要确保更新状态是可见的和原子的-如果一个线程检索到一个对象,而另一个线程在第一个线程可以完成修改和放置之前更新了该对象,该怎么办?它回到 map 上了吗?
T1 -> get() MyObject=30 ------> +1 --------------> put(MyObject=31)
T2 -------> get() MyObject=30 -------> +1 -------> put(MyObject=31)
或者,您也可以使
MyObject
不可变,但是您仍然需要映射
ConcurrentHashMap
,以便其他线程查看
put
-线程缓存行为可能会缓存旧副本,而不刷新并继续重用旧版本。
ConcurrentHashMap
确保其写入对读者可见,并确保线程安全。回顾线程安全的三个先决条件,我们可以从以下方面获得可见性:使用线程安全的数据结构,通过使用不可变对象(immutable对象)获得原子性,最后通过搭载
ConcurrentHashMap
的线程安全性进行排序。
关于java - 解释JIT重新排序的工作方式,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42079959/
这个问题在这里已经有了答案: How does Scala's apply() method magic work? (3 个回答) 9年前关闭。 假设我在 scala 中有一个 MyList 类,其
这个问题在这里已经有了答案: What is a non-capturing group in regular expressions? (18 个回答) Reference - What does
这个问题是针对嵌入式系统的! 我有以下选项来初始化一个对象: Object* o = new Object(arg); 这会将对象放入堆中并返回指向它的指针。我不喜欢在嵌入式软件中使用动态分配。 Ob
我自己搜索过,没能成功的正则表达式。 我有一个 html 文件,其中包含 [] 之间的变量我想把每一个字都写进去。 [client_name][client_company] [cl
我是 Python 新手。我不明白为什么这段代码不起作用: reOptions = re.search( "[\s+@twitter\s+(?P\w+):(?P.*?)\s+]", d
在过去 7 个月左右的时间里,我几乎一直在使用 .NET C# 进行编程。在那之前,我的大部分编程都是用 C++(从学校里学的)。在工作中,我可能需要在接下来的几个月里做一大堆 C 语言。我对 C 的
我是 RE 的新手,我正在尝试获取歌词并分离出歌词标题、和声和主唱: 下面是一些歌词的例子: [Intro] D.A. got that dope! [Chorus: Travis Scott] Ic
这可能是不可能的,但我想检查是否可以用一种简单的方式表达这样的事情: // obviously doesn't work class Foo : IFoo where T: Bar {
我们的应用程序中有“user”和“study”实体,存储在它们各自的表中。一项研究代表一种研究和已收集的数据。它们是多对多的关系,所以我们需要一个链接表:studies_users。 我们为用户分配角
将测试条件添加到 Visual Studio 2010 数据库单元测试(对于 SQL Server 2008)时,这些条件称为例如rowCountCondition1、rowCountConditio
在模拟器上,我可以从设置中卸载 SD 卡。 然后我可以将它安装到我的操作系统上,然后正常卸载它。 我一直无法弄清楚如何在模拟器上重新安装它(无需重新启动)。 提示: adb 命令 remount 是无
假设在一个分支上执行了一系列提交,但该分支尚未与主干重新同步。是否可以从提交中生成全局补丁?是否可以从一系列提交中生成“分组”补丁?如果是,如何? 最佳答案 svn diff -rXXX:YYY UR
在某些情况下,我想在我的应用程序中锁定调整大小功能,为此我尝试对属性进行数据绑定(bind),并且不允许在某些情况下更改它,但没有成功。 有没有办法这样做? 这是我不成功的尝试: XAML: Vie
当我的计算机连接多个显示器时,我可以检测它们,并根据从获取的值设置位置来向它们绘制图形 get(0, 'MonitorPositions') 但是,当我在 MATLAB 运行时断开监视器时,此属性不会
我们有一个grails应用程序,该应用程序在grails数据库中存储了各种域对象。该应用程序连接到第二个数据库,运行一些原始sql,并在表中显示结果。它基本上是一个报告服务器。 我们通过在DataSo
无法比较来自不同容器的迭代器(参见这里的示例: https://stackoverflow.com/a/4664519/225186 )(或者从技术上讲,它不需要有意义。) 这就提出了另一个问题,来自
我有以下情况: 家长 Activity : ParentActivityClass { private Intent intent; @Override public void onCreate(Bu
我经常将元素与附加功能 Hook ,例如: $('.myfav').autocomplete(); $('.myfav').datepicker(); $('.myfav').click(somefu
因此,我将 tooltipster.js 库用于工具提示,并尝试更改工具提示在不同屏幕尺寸上的默认距离。 所以这是默认的 init 的样子: $(inputTooltipTrigger).tool
我在 ARM7 嵌入式环境中工作。我使用的编译器不支持完整的 C++ 功能。它不支持的一项功能是动态类型转换。 有没有办法实现dynamic_cast<>() ? 我使用 Google 寻找代码,但到
我是一名优秀的程序员,十分优秀!