- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
众所周知,如果我们有一些对象引用并且这个引用有 final 字段 - 我们将从 final 字段看到所有可到达的字段(至少在构造函数完成时)
class Foo{
private final Map map;
Foo(){
map = new HashMap();
map.put(1,"object");
}
public void bar(){
System.out.println(map.get(1));
}
}
据我了解,在这种情况下,我们可以保证 bar()
方法始终输出 object
,因为:
1. 我列出了 Foo
类的完整代码, map 是最终的;
<强>2。如果某个线程将看到 Foo
的引用并且此引用 != null,那么我们可以保证从最终 map
引用值可达的值将是实际的。。
我也觉得
class Foo {
private final Map map;
private Map nonFinalMap;
Foo() {
nonFinalMap = new HashMap();
nonFinalMap.put(2, "ololo");
map = new HashMap();
map.put(1, "object");
}
public void bar() {
System.out.println(map.get(1));
}
public void bar2() {
System.out.println(nonFinalMap.get(2));
}
}
这里我们对 bar()
方法有相同的保证,但是 bar2
可以抛出 NullPointerException
尽管发生了 nonFinalMap
赋值在 map
赋值之前。
我想知道 volatile 怎么样:
class Foo{
private volatile Map map;
Foo(){
map = new HashMap();
map.put(1,"object");
}
public void bar(){
System.out.println(map.get(1));
}
}
据我所知,bar()
方法不能抛出 NullPoinerException
但它可以打印 null
; (我完全不确定这方面)
class Foo {
private volatile Map map;
private Map nonVolatileMap;
Foo() {
nonVolatileMap= new HashMap();
nonVolatileMap.put(2, "ololo");
map = new HashMap();
map.put(1, "object");
}
public void bar() {
System.out.println(map.get(1));
}
public void bar2() {
System.out.println(nonFinalMap.get(2));
}
}
我认为这里我们对 bar()
方法有相同的保证 bar2()
不能抛出 NullPointerException
因为 nonVolatileMap
assignment 写了更高的 volatile map assignment 但它可以输出 null
添加在 Elliott Frisch 评论之后
通过比赛示例发布:
public class Main {
private static Foo foo;
public static void main(String[] args) {
new Thread(new Runnable() {
@Override
public void run() {
foo = new Foo();
}
}).start();
new Thread(new Runnable() {
@Override
public void run() {
while (foo == null) ; // empty loop
foo.bar();
}
}).start();
}
}
请证明或更正我对代码片段的评论。
最佳答案
在当前 Java 内存模型领域,volatile
不等于 final
。换句话说,you cannot replace final
with volatile
,并认为安全施工保证是一样的。值得注意的是,这在理论上是可以发生的:
public class M {
volatile int x;
M(int v) { this.x = v; }
int x() { return x; }
}
// thread 1
m = new M(42);
// thread 2
M lm;
while ((lm = m) == null); // wait for it
print(lm.x()); // allowed to print "0"
因此,在构造函数中编写 volatile
字段并不安全。
直觉:在上面的示例中,m
上有一场比赛。通过使 field M.x
volatile
不会消除该竞争,只会使 m
本身 volatile
会有所帮助。换句话说,该示例中的 volatile
修饰符用在了错误的地方。在安全发布中,您必须具有“写入 -> volatile 写入 -> 观察 volatile 写入 -> 读取的 volatile 读取(现在在 volatile 写入之前观察写入)”,而是具有“ volatile 写入 -> 写入 -> 读取 -> volatile 读(不观察 volatile 写)”。
琐事 1: 这个属性意味着我们可以在构造函数中更积极地优化 volatile
-s。这证实了可以放宽未观察到的 volatile 存储(实际上直到具有非转义 this
的构造函数完成后才观察到)的直觉。
知识点 2: 这也意味着您无法安全地初始化 volatile
变量。将上面示例中的 M
替换为 AtomicInteger
,您将获得一种奇特的真实行为!在一个线程中调用 new AtomicInteger(42)
,不安全地发布实例,然后在另一个线程中执行 get()
- 你保证观察到 42
?如前所述,JMM 说“不”。 Java 内存模型的较新修订版努力保证所有初始化的安全构造,以捕获这种情况。还有许多重要的非 x86 端口 have already strengthened this为了安全。
知识问答 3: Doug Lea :“这个 final
vs volatile
问题导致 java.util.concurrent 中的一些曲折结构允许 0 作为基本/默认值.这条规则很糟糕,应该改变。”
也就是说,这个例子可以做得更巧妙:
public class C {
int v;
C(int v) { this.x = v; }
int x() { return x; }
}
public class M {
volatile C c;
M(int v) { this.c = new C(v); }
int x() {
while (c == null); // wait!
return c.x();
}
}
// thread 1
m = new M(42);
// thread 2
M lm;
while ((lm = m) == null); // wait for it
print(lm.x()); // always prints "42"
如果 volatile
字段在 volatile read 观察到构造函数中的 volatile write 写入的值后有传递读取,通常的安全发布规则就会生效。
关于java - volatile 发布保证有多深?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42012658/
我正在编写一个具有以下签名的 Java 方法。 void Logger(Method method, Object[] args); 如果一个方法(例如 ABC() )调用此方法 Logger,它应该
我是 Java 新手。 我的问题是我的 Java 程序找不到我试图用作的图像文件一个 JButton。 (目前这段代码什么也没做,因为我只是得到了想要的外观第一的)。这是我的主课 代码: packag
好的,今天我在接受采访,我已经编写 Java 代码多年了。采访中说“Java 垃圾收集是一个棘手的问题,我有几个 friend 一直在努力弄清楚。你在这方面做得怎么样?”。她是想骗我吗?还是我的一生都
我的 friend 给了我一个谜语让我解开。它是这样的: There are 100 people. Each one of them, in his turn, does the following
如果我将使用 Java 5 代码的应用程序编译成字节码,生成的 .class 文件是否能够在 Java 1.4 下运行? 如果后者可以工作并且我正在尝试在我的 Java 1.4 应用程序中使用 Jav
有关于why Java doesn't support unsigned types的问题以及一些关于处理无符号类型的问题。我做了一些搜索,似乎 Scala 也不支持无符号数据类型。限制是Java和S
我只是想知道在一个 java 版本中生成的字节码是否可以在其他 java 版本上运行 最佳答案 通常,字节码无需修改即可在 较新 版本的 Java 上运行。它不会在旧版本上运行,除非您使用特殊参数 (
我有一个关于在命令提示符下执行 java 程序的基本问题。 在某些机器上我们需要指定 -cp 。 (类路径)同时执行java程序 (test为java文件名与.class文件存在于同一目录下) jav
我已经阅读 StackOverflow 有一段时间了,现在我才鼓起勇气提出问题。我今年 20 岁,目前在我的家乡(罗马尼亚克卢日-纳波卡)就读 IT 大学。足以介绍:D。 基本上,我有一家提供簿记应用
我有 public JSONObject parseXML(String xml) { JSONObject jsonObject = XML.toJSONObject(xml); r
我已经在 Java 中实现了带有动态类型的简单解释语言。不幸的是我遇到了以下问题。测试时如下代码: def main() { def ks = Map[[1, 2]].keySet()
一直提示输入 1 到 10 的数字 - 结果应将 st、rd、th 和 nd 添加到数字中。编写一个程序,提示用户输入 1 到 10 之间的任意整数,然后以序数形式显示该整数并附加后缀。 public
我有这个 DownloadFile.java 并按预期下载该文件: import java.io.*; import java.net.URL; public class DownloadFile {
我想在 GUI 上添加延迟。我放置了 2 个 for 循环,然后重新绘制了一个标签,但这 2 个 for 循环一个接一个地执行,并且标签被重新绘制到最后一个。 我能做什么? for(int i=0;
我正在对对象 Student 的列表项进行一些测试,但是我更喜欢在 java 类对象中创建硬编码列表,然后从那里提取数据,而不是连接到数据库并在结果集中选择记录。然而,自从我这样做以来已经很长时间了,
我知道对象创建分为三个部分: 声明 实例化 初始化 classA{} classB extends classA{} classA obj = new classB(1,1); 实例化 它必须使用
我有兴趣使用 GPRS 构建车辆跟踪系统。但是,我有一些问题要问以前做过此操作的人: GPRS 是最好的技术吗?人们意识到任何问题吗? 我计划使用 Java/Java EE - 有更好的技术吗? 如果
我可以通过递归方法反转数组,例如:数组={1,2,3,4,5} 数组结果={5,4,3,2,1}但我的结果是相同的数组,我不知道为什么,请帮助我。 public class Recursion { p
有这样的标准方式吗? 包括 Java源代码-测试代码- Ant 或 Maven联合单元持续集成(可能是巡航控制)ClearCase 版本控制工具部署到应用服务器 最后我希望有一个自动构建和集成环境。
我什至不知道这是否可能,我非常怀疑它是否可能,但如果可以,您能告诉我怎么做吗?我只是想知道如何从打印机打印一些文本。 有什么想法吗? 最佳答案 这里有更简单的事情。 import javax.swin
我是一名优秀的程序员,十分优秀!