- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
另一个场景,基于之前的问题。在我看来,它的结论足够笼统,对广大读者都有用。引用来自 here 的 Peter Lawrey :
The synchronized uses a memory barrier which ensures ALL memory is in a consistent state for that thread, whether its referenced inside the block or not.
首先,我的问题涉及仅数据可见性。也就是说,原子性(“操作同步”)在我的软件中已经得到保证,所以每个写操作都在对相同值的任何读操作之前完成,反之亦然,等等。因此,问题仅与线程可能缓存的值有关。
考虑 2 个线程,threadA 和 threadB,以及以下类:
public class SomeClass {
private final Object mLock = new Object();
// Note: none of the member variables are volatile.
public void operationA1() {
... // do "ordinary" stuff with the data and methods of SomeClass
/* "ordinary" stuff means we don't create new Threads,
we don't perform synchronizations, create semaphores etc.
*/
}
public void operationB() {
synchronized(mLock) {
...
// do "ordinary" stuff with the data and methods of SomeClass
}
}
// public void dummyA() {
// synchronized(mLock) {
// dummyOperation();
// }
// }
public void operationA2() {
// dummyA(); // this call is commented out
... // do "ordinary" stuff with the data and methods of SomeClass
}
}
已知事实(它们来 self 的软件架构):
operationA1()
和operationA2()
被threadA 调用,operationB()
被调用em>threadBoperationB()
是该类中 threadB 调用的唯一方法。请注意,operationB()
位于同步块(synchronized block)中。operationA1()
, operationB()
, 操作A2()
。保证每个操作在调用前一个操作之前完成。这是由于更高级别的架构同步(消息队列,但现在无关紧要)。正如我所说,我的问题完全与数据可见性相关(即数据副本是最新的还是过时的,例如由于线程自身的缓存)。根据 Peter Lawrey 的引述,operationB()
中的内存屏障确保所有内存在 operationB()< 期间对
。因此,例如如果 threadA 改变了 threadB
处于一致状态operationA1()
中的一些值,这些值将在 threadA 的缓存中写入主内存 code>operationB() 开始。 问题 #1:这是正确的吗?
问题 #2:当 operationB()
离开内存屏障时,operationB()
更改的值(并且可能被 缓存em>threadB
) 将被写回主存。 但是 operationA2() 不会安全 因为没有人要求threadA 与主存同步,对吧?所以
operationB()
的更改现在在主内存中并不重要,因为 threadA 可能仍然有 operationB( )
被调用了。
问题 #3:如果我对 Q.#2 的怀疑是真的,那么再次检查我的源代码并取消注释方法
dummyA()
,并取消注释 dummyA()
在 operationA2()
中调用。我知道这在其他方面可能是不好的做法,但这有什么不同吗?我的(可能是错误的)假设如下:dummyA()
将导致 threadA 从主内存更新其缓存数据(由于 mLock
同步块(synchronized block)),因此它将看到 operationB()
完成的所有更改。也就是说,现在一切都安全了。附带说明一下,方法调用的逻辑顺序如下:
operationA1()
operationB()
dummyA()
operationA2()
我的结论:由于
operationB()
中的同步块(synchronized block),threadB 将看到之前可能已更改的最新数据值 (例如在 operationA1()
中。由于 dummyA()
中的同步块(synchronized block),threadA 将看到在 operationB()
中更改的最新数据副本>。这个思路有没有错误?
最佳答案
您对问题 2 的直觉通常是正确的。在操作 A2 开始时使用 synchronized(mLock) 将发出内存屏障,这将确保操作 A2 的进一步读取将看到操作 B 执行的写入,由于使用 synchronized( mLock) 在操作 B.
但是,要回答问题 1,请注意操作 B 可能看不到操作 A1 执行的任何写入,除非您在操作 A1 的末尾插入一个完整的内存屏障(即,没有任何内容告诉系统从操作 A1 线程的缓存中刷新值) .因此,您可能希望在操作 A1 结束时调用 dummyA。
为了完全安全和更易于维护,并且由于您声明这些方法的执行不会相互重叠,您应该将共享状态的所有操作包含在同步(mLock) block 中而不损失性能。
关于java - 多线程场景下的数据可见性,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11303315/
我在 Mac OsX 10.11 上使用 Xcode 7.0.1 (7A1001) 我使用 carthage 0.9.2 通过以下购物车文件下载reactivecocoa github“Reactiv
我正在将一个对象从属性“模型”(我从 Laravel 中的 Blade 属性模型中获得)分配给数据属性模型。后来数据属性模型发生变化,因为它绑定(bind)到表单输入字段。但 Prop “模型”也发生
当我更新数组内对象的属性然后作为组件的 Prop 传递时,在 svelte 中触发 react 性的正确方法是什么? let items = [{ id: 1, name: 'first'
我是 DRY principle 的坚定拥护者: Every piece of knowledge must have a single, unambiguous, authoritative rep
我正在实现一个需要以下功能的线程: 及时响应终止请求 推送消息 在等待消息时保持对 SendMessage 请求的响应 我对消息泵的初始实现使用了 GetMessage,如下所示: while not
在我的应用程序中,用户获得了一份已到达她的文档列表,并且可以对每个文档执行操作。 文件是分批提交的,当这种情况发生时,列表会增加。这一切都很好,这是预期的行为,但最好有一个按钮“暂停实时数据”,它会忽
我有一个属性为 的数据对象 displaySubtotal 我可以通过以下方式更新该属性的值: data.displaySubtotal = numPad.valueAsAString(); 我的方法
我需要一个垂直 slider 输入。由于内置的 sliderInput 函数无法做到这一点,因此我选择自己实现。根据this thread可以 (I) 使用 CSS 旋转 sliderInput
我正在从自定义用户权限管理系统迁移到 Alanning:roles v2.0 .我有一个非常基本的结构: 基本用户 用户组,每个用户组都有特定的设置。我将它们存储在一个“组”集合中。 管理群组的用户的
Shiny 中的响应式(Reactive)表达式将更改传播到需要去的地方。我们可以使用 isolate 来抑制一些这种行为。 ,但是我们可以抑制基于我们自己的逻辑表达式传播的更改吗? 我给出的例子是一
是否有(或可能有) react 性 Parsec (或任何其他纯函数式解析器)在 Haskell 中? 简而言之,我想逐个字符地为解析器提供数据,并获得与我提供的足够多的结果一样多的结果。 或者更简单
HTML(JADE) p#result Lorem ipsum is javascript j s lo 1 2 4 this meteor thismeteor. meteor input.sear
我有一个被导入函数更改的对象。 https://svelte.dev/repl/e934087af1dc4a25a1ee52cf3fd3bbea?version=3.12.1 我想知道如何使我的更改反
我有一个YUV 420半平面格式的图像,其中字节以这种方式存储: [Y1 Y2 ... [U1 V1.... Yk Yk+1...] Uk' Uk'+1] 其中Y平面的大小是UV平面的两倍,并
如何使用 ReactiveCocoa 订阅从 NSMutableDictionary 添加和删除的对象?另外,我想在它发生变化时广播通知。我的猜测是可以使用 RACMulticastConnectio
我正在构建一个带有多个选项卡的应用程序,其中一些选项卡涉及过多的计算,而另一些选项卡的计算速度很快。一个允许用户在 react 性或手动更新之间进行选择的复选框,与“刷新”按钮结合使用,将是理想的选择
我知道您可以在获取集合时使用 reactive: false 关闭 react 性。如何在内容可编辑区域内的集合字段中实现相同的效果?示例: Template.documentPage.events(
我想在 z3 中表示一个哈希函数,比如 SHA(x)。在做了一些研究之后,似乎 z3 不能很好地支持注入(inject)性,所以我不能有像这样的约束(虽然我意识到这并不是严格意义上的碰撞,但作为一种启
我正在解决一个问题,我想在仪表板中将数据显示为图表(通过 perak:c3 )和表格(通过 aslagle:reactive-table )。我的问题是数据是从 MongoDB 中的集合中提取的,它的
我的 ViewModel 中有这个函数,它返回一个信号,但内部 block 不起作用,我尝试添加断点,但它没有中断。这是我的代码。 func executeLoginAPI() -> RACSigna
我是一名优秀的程序员,十分优秀!