Java中的双重检查(Double

网友投稿 209 2023-07-21


Java中的双重检查(Double

在 Effecitve java 一书的第 48 条中提到了双重检查模式,并指出这种模式在 Java 中通常并不适用。该模式的结构如下所示:

public Resource getResource() {

if (resource == null) {

synchronhttp://ized(this){

if (resource==null) {

resource = new Resource();

}

}

}

return resource;

}

该模式是对下面的代码改进:

public synchronized Resource getResource(){

if (resource == null){

resource = new Resource();

}

return resource;

}

这段代码的目的是对 resource 延迟初始化。但是每次访问的时候都需要同步。为了减少同步的开销,于是有了双重检查模式。

在 Java 中双重检查模式无效的原因是在不同步的情况下引用类型不是线程安全的。对于除了 long 和 double 的基本类型,双重检查模式是适用 的。比如下面这段代码就是正确的:

private int count;

public ibqLySMtsgpnt getCount(){

if (count == 0){

synchronized(this){

if (count == 0){

count = computeCount(); //一个耗时的计算

}

}

}

return count;

}

上面就是关于java中双重检查模式(double-check idiom)的一般结论。但是事情还没有结束,因为java的内存模式也在改进中。Doug Lea 在他的文章中写道:“根据最新的 jsR133 的 Java 内存模型,如果将引用类型声明为 volatile,双重检查模式就可以工作了”。所以以后要在 Java 中使用双重检查模式,可以使用下http://面的代码:

private volatile Resource resource;

public Resource getResource(){

if (resource == null){

synchronized(this){

if (resource==null){

resource = new Resource();

}

}

}

return resource;

}

当然了,得是在遵循 JSR133 规范的 Java 中。

所以,double-check 在 J2SE 1.4 或早期版本在多线程或者 JVM 调优时由于 out-of-order writes,是不可用的。 这个问题在 J2SE 5.0 中已经被修复,可以使用 volatile 关键字来保证多线程下的单例。

public class Singleton {

private volatile Singleton instance = null;

public Singleton getInstance() {

if (instance == null) {

synchronized(this) {

if (instance == null) {

instance = new Singleton();

}

}

}

return instance;

}

}

推荐方法 是Initialization on Demand Holder(IODH),

public class Singleton {

static class SingletonHolder {

static Singleton instance = new Singleton();

}

public static Singleton getInstance(){

return SbqLySMtsgpingletonHolder.instance;

}

}

以上就是本文的全部内容,希望对大家学习java程序设计有所帮助。


版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:Java实现简单邮件发送
下一篇:java中javamail收发邮件实现方法
相关文章

 发表评论

暂时没有评论,来抢沙发吧~