- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
考虑以下代码(乍一看并不完全是这样)。
static class NumberContainer {
int value = 0;
void increment() {
value++;
}
int getValue() {
return value;
}
}
public static void main(String[] args) {
List<NumberContainer> list = new ArrayList<>();
int numElements = 100000;
for (int i = 0; i < numElements; i++) {
list.add(new NumberContainer());
}
int numIterations = 10000;
for (int j = 0; j < numIterations; j++) {
list.parallelStream().forEach(NumberContainer::increment);
}
list.forEach(container -> {
if (container.getValue() != numIterations) {
System.out.println("Problem!!!");
}
});
}
我的问题是:为了绝对确定“问题!!!”不会打印,NumberContainer类中的“value”变量是否需要标记为volatile?
让我解释一下我目前是如何理解这一点的。
在第一个并行流中,NumberContainer-123(比方说)按 ForkJoinWorker-1(比方说)递增。因此 ForkJoinWorker-1 将拥有 NumberContainer-123.value 的最新缓存,即 1。(但是,其他 fork-join worker 将拥有 NumberContainer-123.value 的过时缓存 - 他们将存储值 0。在某些时候,这些其他工作人员的缓存将被更新,但这不会立即发生。)
第一个并行流完成,但公共(public) fork-join 池工作线程未被终止。然后第二个并行流开始,使用非常相同的公共(public) fork-join 池工作线程。
现在假设,在第二个并行流中,递增 NumberContainer-123 的任务分配给 ForkJoinWorker-2(比方说)。 ForkJoinWorker-2 将拥有自己的缓存值 NumberContainer-123.value。如果在 NumberContainer-123 的第一次和第二次增量之间经过了很长一段时间,那么 ForkJoinWorker-2 的 NumberContainer-123.value 缓存可能是最新的,即值 1 将被存储,并且一切都是好的。但是,如果 NumberContainer-123 非常短,第一次和第二次增量之间耗时会怎样?那么可能 ForkJoinWorker-2 的 NumberContainer-123.value 缓存已经过时,存储了值 0,导致代码失败!
我上面的描述是否正确?如果是这样,谁能告诉我两次递增操作之间需要什么样的时间延迟才能保证线程之间的缓存一致性?或者,如果我的理解有误,那么有人可以告诉我是什么机制导致线程局部缓存在第一个并行流和第二个并行流之间被“刷新”吗?
最佳答案
它应该不需要任何延迟。当您离开 ParallelStream
的 forEach
时,所有任务都已完成。这在 forEach
的增量和结束之间建立了一个happens-before 关系。所有 forEach
调用都按从同一线程调用的顺序进行排序,同样地,检查发生在所有 forEach
调用之后。
int numIterations = 10000;
for (int j = 0; j < numIterations; j++) {
list.parallelStream().forEach(NumberContainer::increment);
// here, everything is "flushed", i.e. the ForkJoinTask is finished
}
回到你关于线程的问题,这里的诀窍是,线程是无关紧要的。内存模型取决于happens-before 关系,而 fork-join 任务确保 happens-before 调用 forEach
和操作体,以及操作体和 forEach
的返回值之间(即使返回值是 Void
)
另见 Memory visibility in Fork-join
正如@erickson 在评论中提到的,
If you can't establish correctness through happens-before relationships, no amount of time is "enough." It's not a wall-clock timing issue; you need to apply the Java memory model correctly.
此外,从“刷新”内存的角度来考虑它是错误的,因为还有更多的事情会影响你。例如,冲洗是微不足道的:我没有检查过,但可以打赌任务完成时只有内存障碍;但是你可能会得到错误的数据,因为编译器决定优化非 volatile 读取(变量不是 volatile 的,并且在这个线程中没有改变,所以它不会改变,所以我们可以将它分配给一个寄存器,et voila),以 happens-before 关系允许的任何方式重新排序代码,等等。
最重要的是,所有这些优化都可以并且会随着时间的推移而改变,所以即使您转到生成的程序集(可能会因加载模式而异)并检查所有内存屏障,也不能保证您的代码会工作除非你能证明你的读取发生在你的写入之后,在这种情况下,Java 内存模型就在你这一边(假设 JVM 中没有错误)。
至于巨大的痛苦,ForkJoinTask
的目标就是让同步变得微不足道,所以尽情享受吧。它(似乎)是通过将 java.util.concurrent.ForkJoinTask#status
标记为易变的来完成的,但这是您不应该关心或依赖的实现细节。
关于Java - 连续并行流之间的缓存一致性?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52009032/
GhostScript PDF/A 生成好像有错误 当您通过 GhostScript 生成 PDF/A 文档时,当您单击 Adobe Reader 时,会出现一个一致性选项卡,其中显示: “一致性
我有一个需要测试的 XPath 引擎实现。 是否有一组标准的一致性测试可以用来验证是否符合 XPath 规范(与 XSLT 相关)。 什么将是完美的 XML 文档 XPath 表达式和预期的结果。 最
帮助我了解在这种情况下我可以期望与 MongoDB 的一致性级别。 我们正在运行一个副本集,其中 Mongoid 中的 consistency 标志设置为 strong,这意味着只读到 master。
假设我有一个采用一个参数的方法。 此参数应满足以下要求: 'of type':方法需要知道参数属于特定类(或子类)。 'implements interface':方法需要知道参数实现了特定的接口(i
当协议(protocol)将属性声明为可选而具体类型将其声明为非可选时,如何使具体类型符合协议(protocol)? 这是问题所在: protocol Track { var trackNum
我正在考虑使用浏览器的 navigator.mimeTypes 数组作为第三级用户/浏览器标识符。例如,当我在 Chrome 上运行时... console.log(navigator.mimeTyp
我有以下协议(protocol): protocol ProtoAInput { func funcA() } protocol ProtoA { var input: ProtoAI
如果选择“最终”一致性,则发生写入的区域内的一致性是什么? 如果我只需要区域强一致性,应该选择哪个选项? 最佳答案 如果您需要在主要区域内进行强读取,则应该选择强一致性或有界过时一致性。 关于azur
您好,我是一名初学者,目前正在尝试学习 java 编程。课本上的问题: 编写一个程序来帮助人们决定是否购买混合动力汽车。你的程序的输入应该是:•新车的成本•预计每年行驶里程•预计汽油价格 •每加仑英里
我正在尝试制作一个可以在 UILabel 上使用的 Swift 协议(protocol), UITextField , 和 UITextView包含他们的text , attributedText ,
我有一个类扩展: extension UICollectionViewCell { class func registerFromNibInCollectionView(collectionV
为了在 Swift 中模拟对象进行测试,我通常遵循这样的模式:编写一个协议(protocol)来描述我想要的对象的行为,然后使用 Cuckoo 为其生成模拟以进行测试。 通常,这些协议(protoco
假设我有两个非通用协议(protocol)(1) protocol StringValue { var asString: String {get} } protocol StringProv
我有一组协议(protocol)可以在 UITableView 中显示一个元素: protocol TableRepresentableRow { var title: String { get
关闭。这个问题是not reproducible or was caused by typos .它目前不接受答案。 这个问题是由于错别字或无法再重现的问题引起的。虽然类似的问题可能是on-topi
用“class”标记 CacheManager 解决了我的问题。 案例:一个简单的缓存器,mutating get 不是我想要的,那么对于引用类型或类类型应该怎么做? protocol Cacher
我想要一个符合协议(protocol)的变量,但是 swift 编译器告诉我协议(protocol)没有确认。 protocol A {} protocol B { var a : A { g
如果我有一个类 Christmas 和一个协议(protocol) Merry,要使 Christmas 符合 Merry,很多人会这样做: class Christmas { ... } e
@objc public protocol P1 { func p1foo() } @objc public protocol P2 { func p2foo() } class A: NSO
我有一些结构符合的基本协议(protocol)(模型)。它们也符合 Hashable protocol Model {} struct Contact: Model, Hashable { v
我是一名优秀的程序员,十分优秀!