- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
如果我有一个在线程之间共享的对象,在我看来,每个字段都应该是 final
或 volatile
,理由如下:
如果该字段应该改变(指向另一个对象,更新原始值),那么该字段应该是volatile
,以便所有其他线程对新值进行操作。仅仅对访问所述字段的方法进行同步是不够的,因为它们可能会返回缓存值。
如果该字段永远不应该更改,请将其设为 final
。
但是,我找不到任何关于此的信息,所以我想知道这个逻辑是否有缺陷或太明显了?
EDIT 当然可以使用 final AtomicReference
或类似的,而不是 volatile。
编辑 例如,参见 Is getter method an alternative to volatile in Java?
EDIT以避免混淆:这个问题是关于缓存失效的!如果两个线程对同一个对象进行操作,对象的字段可以被缓存(每个线程) ,如果它们没有被声明为 volatile。如何保证缓存正确失效?
最终编辑感谢@Peter Lawrey,他向我指出了 JLS §17(Java 内存模型)。据我所知,它声明同步在操作之间建立了先发生关系,因此如果这些更新“发生在之前”,则一个线程可以看到来自另一个线程的更新,例如如果非 volatile 字段的 getter 和 setter 已同步
。
最佳答案
虽然我觉得 private final
可能应该是字段和变量的默认值,使用 var
之类的关键字使其可变,在不需要时使用 volatile是
final
不同,它通过说不应该更改来提高清晰度,在不需要时使用 volatile
,可能会在读者尝试工作时造成混淆找出为什么它会变得不稳定。if the field should be changed (point to another object, update the primitive value), then the field should be volatile so that all other threads operate on the new value.
虽然这对于读取来说很好,但请考虑这个简单的情况。
volatile int x;
x++;
这不是线程安全的。因为它是一样的
int x2 = x;
x2 = x2 + 1; // multiple threads could be executing on the same value at this point.
x = x2;
更糟糕的是,使用 volatile
会使此类 bug 更难找到。
正如 yshavit 指出的那样,使用 volatile
更新多个字段更难解决,例如HashMap.put(a, b)
更新多个引用。
Merely a synchronization on the methods which access said field is insufficient because they might return a cached value.
synchronized 为您提供 volatile
的所有内存保证以及更多,这就是它明显变慢的原因。
注意:仅仅 synchronized
-ing 每个方法也并不总是足够的。 StringBuffer
使每个方法都同步,但在多线程上下文中比无用更糟糕,因为它的使用可能容易出错。
很容易假设实现线程安全就像撒仙尘一样,添加一些神奇的线程安全,你的错误就会消失。问题是线程安全更像是一个有很多洞的桶。堵住最大的洞,bug 似乎就消失了,但除非你把它们都堵上,否则你没有线程安全,但它可能更难找到。
就同步与易失而言,这表明
Other mechanisms, such as reads and writes of volatile variables and the use of classes in the java.util.concurrent package, provide alternative ways of synchronization.
https://docs.oracle.com/javase/specs/jls/se7/html/jls-17.html
关于Java:使所有字段成为最终字段或可变字段?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53738662/
为什么禁用类型像 type t = A of int | B of string * mutable int 虽然允许此类类型: type t = A of int | B of string * i
我正在寻找一种类似结构的数据结构,我可以从中创建多个实例并具有某种类型提示而不是不可变的。 所以我有这样的东西: class ConnectionConfig(NamedTuple): nam
我需要转到引用的结构: class SearchKnot { var isWord : Bool = false var text : String = "" var to
如sec 10.4.3中所述 当控制进入执行时,执行以下步骤 功能对象F(调用者)中包含的功能代码的上下文 提供thisArg,而调用方提供argumentsList: 如
i make a game that start display Activity indicator And activity indicator bottom display UiLable wi
编辑:我在这里不断获得支持。只是为了记录,我认为这不再重要。自从我发布它以来我就不再需要它了。 我想在 Scala 中执行以下操作... def save(srcPath: String, destP
使用可变对象作为 Hashmap 键是一种不好的做法吗?当您尝试使用已修改足以更改其哈希码的键从 HashMap 中检索值时,会发生什么? 例如,给定 class Key { int a; /
如果您在Kotlin中访问List类型的Java值,则将获得(Mutable)List!类型。 例如。: Java代码: public class Example { public stati
我编写了 str 类(内置)的以下扩展,以便执行以下操作:假设我有字符串 "Ciao" ,通过做"Ciao" - "a"我想要的结果是字符串 "Cio" 。这是执行此操作的代码,并且运行良好: cla
使用可变对象作为 Hashmap 键是一种不好的做法吗?当您尝试使用已修改足以更改其哈希码的键从 HashMap 中检索值时,会发生什么? 例如,给定 class Key { int a; /
我正在为我的公司设计一个数据库来管理商业贷款。每笔贷款都可以有担保人,可以是个人或公司,在借款业务失败时作为财务支持。 我有 3 个表:Loan、Person 和 Company,它们存储明显的信息。
我使用二进制序列化从 C# 类中保存 F# 记录。一切正常: F#: type GameState = { LevelStatus : LevelStatus
import javax.swing.JOptionPane; public class HW { public static void main(String[] args) { Strin
使用 flatbuffer mutable 有多少性能损失? 是否“正确”使用 FlatBuffers 来拥有一个应该可编辑的对象/结构(即游戏状态) 在我的示例中,我现在有以下类: class Ga
std::function create_function (args...) { int x = initial_value (args...); return [x] () mut
我需要在 for 循环中找到用户输入的字符。我通常会这样做 如果(句子[i] == 'e') 但是因为在这里,'e' 将是一个单字母字符变量,我不知道如何获取要比较的值。我不能只输入 if (sent
我有一个这样的算法: let seed: Foo = ... let mut stack: Vec = Vec::new(); stack.push(&seed); while let Some(ne
这个问题可能看起来非常基础,但我很难弄清楚如何做。我有一个整数,我需要使用 for 循环来循环整数次。 首先,我尝试了—— fn main() { let number = 10; // An
如果我有以下结构: struct MyStruct { tuple: (i32, i32) }; 以及以下函数: // This will not compile fn function(&mut s
我希望在每个 session 的基础上指定列的默认值。下面的脚本不起作用,但描述了我想如何使用它。我目前使用的是 MySQL 5.5.28,但如果需要可以升级。 CREATE TABLE my_tbl
我是一名优秀的程序员,十分优秀!