- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我正在使用 joda,因为它在多线程方面享有盛誉。它大大提高了多线程日期处理的效率,例如通过使所有 Date/Time/DateTime 对象不可变。
但在这种情况下,我不确定 Joda 是否真的在做正确的事情。可能是这样,但我很想看到解释。
当调用 DateTime 的 toString() 时,Joda 会执行以下操作:
/* org.joda.time.base.AbstractInstant */
public String toString() {
return ISODateTimeFormat.dateTime().print(this);
}
所有格式化程序都是线程安全的(它们也是不可变的),但是格式化程序工厂是什么:
private static DateTimeFormatter dt;
/* org.joda.time.format.ISODateTimeFormat */
public static DateTimeFormatter dateTime() {
if (dt == null) {
dt = new DateTimeFormatterBuilder()
.append(date())
.append(tTime())
.toFormatter();
}
return dt;
}
这是单线程应用程序中的常见模式,但众所周知,它在多线程环境中容易出错。
我看到了以下危险:
没问题,因为这只是一个辅助对象(与正常的单例模式情况不同),一个保存在 dt 中,另一个丢失,迟早会被垃圾回收。
(在说我疯之前,请阅读此 Wikipedia article 中的类似情况。)
那么 Joda 如何确保没有部分创建的格式化程序发布到这个静态变量中?
感谢您的解释!
返回
最佳答案
你说过,格式化程序是只读的。如果他们仅使用最终字段(我没有阅读格式化程序源代码),那么在 Java 语言规范的第 3 版中,他们会受到“最终字段语义”的部分对象创建的保护。我没有检查第 2 个 JSL 版本,也不确定该版本中的初始化是否正确。
查看 JLS 中的第 17.5 和 17.5.1 章。我将为所需的发生前关系构建一个“事件链”。
首先,在构造函数的某处有一个写入格式化程序中的最终字段。就是写w。当构造函数完成时,将执行“卡住”操作。我们称它为f。在程序顺序后面的某个地方(从构造函数返回后,可能是其他一些方法并从 toFormatter 返回)有一个写入 dt 字段。让我们给这个写一个名字。此写入 (a) 在“程序顺序”(单线程执行顺序)中的卡住操作 (f) 之后,因此 f happens-before a (hb(f, a)) 只是根据 JLS 定义。呼,初始化完成...:)
有时,在另一个线程中,会调用 dateTime().format。那时我们需要两次读取。两者中的第一个是读取格式化程序对象中的最终变量。我们称它为 r2(为了与 JLS 一致)。两者中的第二个是格式化程序的“this”读取。在读取 dt 字段时调用 dateTime() 方法期间会发生这种情况。我们称其为 read r1。我们现在有什么?读取 r1 看到一些写入 dt。我认为写入是上一段中的操作 a(为了简单起见,只有一个线程写入该字段)。由于 r1 看到写 a,则有 mc(a, r1)(“内存链”关系,第一个子句定义)。当前线程没有初始化格式化程序,在操作 r2 中读取它的字段并看到在操作 r1 中读取的格式化程序的“地址”。因此,根据定义,有一个 dereferences(r1, r2)(另一个从 JLS 排序的 Action )。
我们在卡住前写入,hb(w, f)。我们在分配 dt 之前卡住,hb(f, a)。我们读取了 dt, mc(a, r1)。我们在 r1 和 r2 之间有一个解引用链,dereferences(r1, r2)。根据 JLS 的定义,所有这些都会导致 hb(w, r2) 发生之前的关系。此外,根据定义,hb(d, w) 其中 d 是对象中最终字段的默认值写入。因此,读取 r2 看不到写入 w 而必须看到写入 r2(程序代码中唯一对该字段的写入)。
更多间接字段访问的顺序相同(存储在最终字段中的对象的最终字段,等等)。
但这还不是全部!无法访问部分构造的对象。但是还有一个更有趣的错误。在缺少任何显式同步的情况下,dateTime() 可能会返回 null。我不认为在实践中可以观察到这种行为,但 JLS 第 3 版不会阻止这种行为。第一次读取方法中的 dt 字段可能会看到另一个线程初始化的值,但第二次读取 dt 会看到“写入默认值”。没有 happens-before 关系可以阻止它。这种可能的行为特定于第 3 版,第 2 版有“写入主内存”/“从主内存读取”,这不允许线程看到及时返回的变量值。
关于java - 部分构造的对象/多线程,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2509847/
谁能解释一下原因: (define a (lambda() (cons a #f))) (car (a)) ==> procedure ((car (a))) ==> (procedure . #f)
这是 PyBrain 网站的摘录。我了解大部分正在发生的事情,但是一行让我完全难住了。我以前从未在 python 代码中看到过这样的东西。这是整个循环,对于上下文: for c in [0,
我是gradle / groovy的新手。我想创建将做一些事情的自定义任务。我的第一个问题是任务完成时该如何做?我可以覆盖doFirst / doLast闭包吗?也许我可以重写某些在开始和结束时都会执
我刚刚开始评估 MS 企业库。他们使用以下指令来获取实例: var customerDb = EnterpriseLibraryContainer.Current.GetInstance("C
这是我的 if else Ansible 逻辑.. - name: Check certs exist stat: path=/etc/letsencrypt/live/{{ rootDomain
我正在使用construct 2.8 对一些失传已久的 Pascal 程序创建的一些文件的 header 进行逆向工程。 header 由许多不同的记录组成,其中一些是可选的,我不确定顺序是否固定。
我在将 getchar() 的输入放入 char *arr[] 数组时遇到问题。我这样做的原因是因为输入数据(将是一个带有命令行参数的文件)将存储在一个 char 指针数组中以传递给 execvp 函
通常我们不能约束类型参数 T派生自密封类型(例如 struct 类型)。这将毫无意义,因为只有一种类型适合,因此不需要泛型。所以约束如下: where T : string 或: where T :
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 9 年前。 Improve th
#include using namespace std; class A { private: int m_i; friend int main(int argc, char cons
这个问题在这里已经有了答案: Are there legitimate uses for JavaScript's "with" statement? (33 个答案) 关闭 9 年前。 我有这个代
在this answer我看到了下一个 Bash 结构。 yes "$(< file.txt)" 什么意思 "$(< file.txt)" ? 我明白了 命令替换 - $(command)用命令的结
if (a == 1) //do something else if (a == 2) //do something else if (a == 3) //do somethi
关于构造的快速简单的问题。 我有以下用于将项目添加到 ListView 的代码。 ListViewItem item = new ListViewItem(); item.Text = file; i
我想使用 std::vector 来控制给定的内存。首先,我很确定这不是好的做法,但好奇心占了上风,无论如何我都想知道如何做到这一点。 我遇到的问题是这样的方法: vector getRow(unsi
下面显示了一段简单的javascript: var mystring = ("random","ignored","text","h") + ("ello world") 这个字符串会生成 hello
在 Java 中,创建对象的标准方法是使用 MyClass name = new MyClass(); 我也经常看到构造 new MyClass() { /*stuff goes in here*/
我正在编写 C++ ndarray 类。我需要动态大小和编译时大小已知的数组(分别分配自由存储和分配堆栈)。我想支持从嵌套的 std::initializer_list 进行初始化。 动态大小的没问题
我正在将一个项目从 Visual Studio 2005 转换为 Visual Studio 2008,并提出了上述结构。 using Castle.Core.Resource; using Cast
我想知道我在这里的想法是否正确,我主要针对接口(interface)进行编程,所以我想知道下面的类是否应该通过 DI 注入(inject),或者我应该自己实例化一个类... 注意:这些服务保存在我的核
我是一名优秀的程序员,十分优秀!