via: http://blog.csdn.net/natee/article/details/4408245
Singleton模式主要作用是保证在Java应用程序中,一个类Class只有一个实例存在。
第一种形式
public class Singleton {
// Early initialization.定义的时候就初始化(不推荐)。
private static Singleton s = new Singleton();
private Singleton() {
}
public static Singleton getInstance() {
return s;
}
}
第二种形式
public class Singleton {
private static Singleton s;
private Singleton() {
}
// Lazy initialization. 延迟初始化,使用的时候再初始化,效率较高(推荐)。
public static Singleton getInstance() {
if (s == null) {
s = new Singleton();
}
return s;
}
}
实现的关键:
1. 将所有的构造函数都设为private ,而且必须显示的指定构造函数(不能设置为默认的,因为默认构造函数是package访问权限)。
2. 注意clone()方法。
例如, 如果基类实现了cloneable接口的话,子类就应该重写该方法。当然,在应用中应该灵活运用各种方法来防止clone()的各种情况。
@Override
protected Object clone() throws CloneNotSupportedException {
throw new CloneNotSupportedException();
}
多线程调用singleton方法:
如果在网络编程中,要注意多线程访问singleton引发的一系列问题:
public class Singleton {
private static Singleton s;
private Singleton() {
}
// 如果多个线程同时访问, 有可能会出现多个实例。
public static Singleton getInstance() {
// 第一次初始化时,多个线程同时执行"if (s == null)",判断结果都为真,所以都会执行下面的操作:"s = new Singleton()",由此引发多个实例的出现。
if (s == null) {
s = new Singleton();
}
return s;
}
}
解决方法1(不推荐):
public class Singleton {
private static Singleton s;
private Singleton() {
}
// 将该方法加上synchronized关键字。这种方法确实能解决问题,但效率不是很高。因为每次调用该方法的时候,都需要阻塞该方法,但事实上只有第一次初始化的时候才会出现这种情况,所以......
public static synchronized Singleton getInstance() {
if (s == null) {
s = new Singleton();
}
return s;
}
}
解决方法2(推荐):
public class Singleton {
private static Singleton singleton;
private Singleton() {
}
public static Singleton getInstance() {
if (singleton == null) {
synchronized (Singleton.class) {
if (singleton == null) {
singleton = new Singleton();
}
}
}
return singleton;
}
}