IT之家Android知识程序员

单例模式以及synchronized用法详解

2017-07-04  本文已影响115人  搅局者

单例模式理解

单例模式是指确保对于一个给定的类只有一个实例存在,这个实例有一个全局唯一的访问点。它通常采用懒加载的方式(通过判空处理,进行延迟加载)在第一次用到实例的时候再去创建它。

public class SingletonClass{

private static volatile SingletonClass instance = null;

public static SingletonClass getInstance(){

if(instance == null){  //第一次判空

synchronized(SingletonClass.class){

if(instance == null){  //为什么进行第二次判空?

instance = new SingletonClass();

}

}

}

return instance;

}

private SingletonClass(){        }

}

解释单例模式中的一些疑惑

(1)为什么用同步锁?

为了避免在 APP 中,多线程竞争同一资源的时候,出现数据错乱和安全的问题。我们需要给共享的资源加锁,让线程一个个地通过,以确保每次线程读取的数据是正确性和安全性。

(2)为什么懒汉模式中要用两个if()判断对象是否为空?

第一个if判断,大家应该可以理解。关键是第二个if判断,大家会想第一个if不是已经判断出对象是否为空了吗? 现在我拿上面写的例子来解释一下,当x事件和y事件同时调用SingletonClass的对象时,如果此时instance为null,那么x 和 y 都会经过第一个if判断,到达同步锁那里。其中一个(假设 x)进入到里面以后,y在锁(synchronized())外等候,x从锁里出来后,instance已经不为null,这时y进入锁后,如果没有if判断instance是否为空,那么instance就会又执行了一次new操作,这样x 事件和 y事件调用的就不是同一个对象了,这样就失去了单例模式的意义,所以要进行两次if判断。

(3)synchronized作用?

作用:当它用来修饰一个方法或者一个代码块的时候,能够保证在同一时刻最多只有一个线程执行该段代码。

注意:在用synchronized关键字的时候,能缩小代码段的范围就尽量缩小,能在代码段上加同步就不要再整个方法上加同步。这样可以使代码更大程度的并发,否则,锁的代码段太长了,别的线程要等很久,会影响应用的性能。

(4)synchronized和static synchronized区别?

synchronized是对类的当前实例(当前对象)进行加锁,防止其他线程同时访问该类的当前实例的所有synchronized代码块。而static synchronized是要控制类的所有实例的并发访问,限制多线程中该类的所有实例同时访问jvm中该类所对应的代码块。

实际上,在类中如果某方法或某代码块中有synchronized,那么在生成一个该类实例后,该实例也就有一个监视块,防止线程并发访问该实例的synchronized保护块,而static synchronized则是所有该类的所有实例公用的一个监视块,这就是他们两个的区别。

上一篇下一篇

猜你喜欢

热点阅读