Java中的双重检查(Double-Check)详解
在EffecitveJava一书的第48条中提到了双重检查模式,并指出这种模式在Java中通常并不适用。该模式的结构如下所示:
publicResourcegetResource(){
if(resource==null){
synchronized(this){
if(resource==null){
resource=newResource();
}
}
}
returnresource;
}
该模式是对下面的代码改进:
publicsynchronizedResourcegetResource(){
if(resource==null){
resource=newResource();
}
returnresource;
}
这段代码的目的是对resource延迟初始化。但是每次访问的时候都需要同步。为了减少同步的开销,于是有了双重检查模式。
在Java中双重检查模式无效的原因是在不同步的情况下引用类型不是线程安全的。对于除了long和double的基本类型,双重检查模式是适用的。比如下面这段代码就是正确的:
privateintcount;
publicintgetCount(){
if(count==0){
synchronized(this){
if(count==0){
count=computeCount();//一个耗时的计算
}
}
}
returncount;
}
上面就是关于java中双重检查模式(double-checkidiom)的一般结论。但是事情还没有结束,因为java的内存模式也在改进中。DougLea在他的文章中写道:“根据最新的JSR133的Java内存模型,如果将引用类型声明为volatile,双重检查模式就可以工作了”。所以以后要在Java中使用双重检查模式,可以使用下面的代码:
privatevolatileResourceresource;
publicResourcegetResource(){
if(resource==null){
synchronized(this){
if(resource==null){
resource=newResource();
}
}
}
returnresource;
}
当然了,得是在遵循JSR133规范的Java中。
所以,double-check在J2SE1.4或早期版本在多线程或者JVM调优时由于out-of-orderwrites,是不可用的。这个问题在J2SE5.0中已经被修复,可以使用volatile关键字来保证多线程下的单例。
publicclassSingleton{
privatevolatileSingletoninstance=null;
publicSingletongetInstance(){
if(instance==null){
synchronized(this){
if(instance==null){
instance=newSingleton();
}
}
}
returninstance;
}
}
推荐方法是InitializationonDemandHolder(IODH),
publicclassSingleton{
staticclassSingletonHolder{
staticSingletoninstance=newSingleton();
}
publicstaticSingletongetInstance(){
returnSingletonHolder.instance;
}
}
以上就是本文的全部内容,希望对大家学习java程序设计有所帮助。