- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有一个 String
字段,该字段被初始化为 null
但随后可能被多个线程访问。该值将在第一次访问时延迟初始化为幂等计算的值。
该字段是否需要 volatile
才能保证线程安全?
这是一个例子。
public class Foo {
private final String source;
private String BAR = null;
public Foo(String source) {
this.source = source;
}
private final String getBar() {
String bar = this.BAR;
if (bar == null) {
bar = calculateHashDigest(source); // e.g. an sha256 hash
this.BAR = bar;
}
return bar;
}
public static void main(String[] args) {
Foo foo = new Foo("Hello World!");
new Thread(() -> System.out.println(foo.getBar())).start();
new Thread(() -> System.out.println(foo.getBar())).start();
}
}
我使用了System.out.println()
作为示例,但我并不担心它的调用互锁时会发生什么。 (尽管我很确定这也是线程安全的。)
BAR
是否需要 volatile
?
我认为答案是否, volatile
不是必需的,并且是它是线程安全的,主要是因为这个excerpt from JLS 17.5:
final
fields also allow programmers to implement thread-safe immutable objects without synchronization. A thread-safe immutable object is seen as immutable by all threads, even if a data race is used to pass references to the immutable object between threads.
并且由于 String
的 char value[]
字段确实是 final
。
(int hash
不是 final
,但它的延迟初始化看起来也不错。)
编辑:进行编辑以澄清用于 BAR
的值是固定值。它的计算是幂等的并且没有副作用。我不介意计算是否跨线程重复,或者 BAR 是否由于内存缓存/可见性而有效地成为线程本地的。我担心的是,如果它非空,那么它的值是完整的,而不是部分的。
最佳答案
您的代码(技术上)不是线程安全的。
String
确实是一个正确实现的不可变类型,并且您关于其 final
字段的说法是正确的。但这不是线程安全问题的所在。
第一个问题是 BAR
的延迟初始化中存在竞争条件。如果两个线程同时调用 getBar()
,它们都会将 BAR
视为 null
,然后都尝试对其进行初始化。
第二个问题是存在内存危险。由于一个线程对 BAR
的写入与另一个线程对 BAR
的后续读取之间不存在发生之前关系,因此无法保证第二个线程线程将看到BAR
的初始化值。因此,它可能会重复初始化。
请注意,在示例中所写,这两个问题不是实际的线程安全问题。您正在执行的初始化是幂等的。多次初始化 BAR
对代码的行为没有影响,因为您始终将其初始化为对同一 String
对象的引用。 (单个冗余初始化的成本太小,无需担心。)
但是,如果 BAR
是对可变对象的引用,或者初始化成本很高,那么这就是一个真正的线程安全问题。
正如 @Ravindra 所说,简单的解决方案是将 getBar
声明为 synchronized
。这解决了这两个问题。
您声明 BAR
的想法解决了内存危险,但没有解决竞争条件。
您在问题中添加了以下内容:
Edit to clarify the value intended for
BAR
is a fixed value. Its calculation is idempotent and has no side-effects. I don't mind if the calculation is repeated across threads, or ifBAR
becomes effectively a thread-local due to memory-caching / visibility. My concern is, if it's non-null then it's value is complete and not somehow partial.
这并没有改变我上面所说的。如果该值是一个String
,那么它就是一个正确实现的不可变对象(immutable对象),并且您将始终看到一个完整的值而不管任何其他内容。 JLS 引言就是这么说的!
(实际上,我掩盖了 String
使用非 final
字段来保存延迟计算的哈希码的细节。但是, String::hashCode
实现会处理这个问题。它不存在线程安全问题。如果您愿意,请自行检查。)
关于java - 通过引用延迟初始化的非 volatile 字符串进行访问是线程安全的吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55633948/
在 Windows 世界中,什么是正确的名称。具有导出函数的老式 C++ DLL?不是 COM DLL,也不是 .NET DLL。我们以前通过调用 LoadLibrary() 和 GetProcAdd
目前我正在使用javaEE7,我有一个场景如下。在我的 JSF Web 应用程序中,我有一个事件监听器(不是 JSF 事件),当事件调用时,它会执行某些操作,然后将这些信息更新到我的 Web 应用程序
这不是 AJAX 请求/响应回调问题... 我正在使用 Dojo 1.5 构建网格。我正在尝试 dojo.connect具有功能的扩展/收缩按钮。我的问题是 grid.startup()在创建实际 D
非 Webkit Opera 是 very specific在某些功能中,因此通常通过 JavaScript 检测到 the following way . 但是,Opera Next 几乎是 Goo
我已查看以下链接中给出的所有日志,但未能找到 IP 地址: https://developer.couchbase.com/documentation/server/3.x/admin/Misc/Tr
我有一个命令行程序,它根据一组源文件生成一个我想在我的 Android gradle 构建 (A) 中使用的 jar 文件。这个命令行程序只是将一个 jar 文件存储在磁盘上的一个目录中。 我如何创建
下面的 htaccess 命令将所有非 www 转移到 http www RewriteEngine On RewriteCond %{HTTP_HOST} !^www\. RewriteRule ^
我正在使用自定义链接器脚本将内核镜像分为两部分。第一个是普通代码和数据,第二个是初始化代码和不再需要时将被丢弃的数据。初始化部分也不像内核本身那样在地址空间之间共享,因此如果 fork() 仍然存在(
这个问题在这里已经有了答案: Several unary operators in C and C++ (3 个答案) What is the "-->" operator in C++? (29
假设我有一个类设置如下: class A { public: virtual void foo() { printf("default implementation\n"); } }; c
#include using namespace std; int main(int argc, char *argv[]) { int i=-5; while(~(i)) {
近期,百度搜索引擎变化无常,很多企业站、行业站、门户站、论坛等站点遭到了降权,特别是比比贴分类信息网直接遭到了拔毛,这对于广大站长来说是一种打击,也是各个企业、行业的打击。 至今,很多网站已经恢复
我现在正在使用 IBM TPM v1332 + IBM TSS v1470 并尝试将一些基本关键字/密码存储到 TPM 上的非 volatile 内存中。我找到了两种方法。一种是创建一个密封对象并使用
我的 PHP 脚本中有一个正则表达式,如下所示: /(\b$term|$term\b)(?!([^)/iu 这与 $term 中包含的单词匹配,只要前后有单词边界并且它不在 HTML 标记内即可。 但
我想显示用户名称地址(请参阅 www.ipchicken.com ),但我唯一能找到的是 IP 地址。我尝试了反向查找,但也没有用: IPAddress ip = IPAddress.Parse(th
只有 UI 线程能够显示到屏幕上,还是其他线程也可以这样做? 最佳答案 不,您只能直接从 UI 线程访问 UI,但您可以编码来自其他线程的结果,例如使用 Control.Invoke 或 contro
我正在使用现代 Excel 滚动条(不是旧的 ActiveX 类型,即开发人员 > 插入 > 表单控件 > 滚动条)并且想检测它的值何时更改。我找不到有关此类对象的更改事件的任何信息。您可以在单击时分
当我使用这段代码时 IE 6 确实正确使用了指定的样式表,但所有其他浏览器在应该使用基本上声明的样式表时会忽略这两种样式表,如果您不是 IE,请使用此样式表。 有什么想法吗? 最佳答案 n
我想指定 2 mssql 表之间的关系。 付款类别和付款。 paymentcategory.id 加入 payout.category 列。 在 payout.json 模型中 我指定为外键:id,
我正在尝试制作非 volatile UDF,但似乎不可能。因此,这是我非常简单的test-UDF: Option Explicit Dim i As Integer Sub Main() i = 0
我是一名优秀的程序员,十分优秀!