单例模式,大家都了解,不过如果没有系统地看看书,可能也不能很规范地说出单例的几种方法。
1.同步延迟加载
public class Singleton {
private static Singleton uniqBean;
private Singleton() {
}
public static synchronized Singleton getInstance() {
if (uniqBean == null) {
uniqBean = new Singleton();
}
return uniqBean;
}
}
如上代码,如果去掉synchronized,在多线程编程环境下就会有问题。而加上之,在多线程情况下会导致严重的性能问题,因为每个线程都要等其他线程执行完getInstance。所以,很不推荐使用这个方式。
2.急切初始化
public class Singleton {
private static Singleton uniqBean = new Singleton();
private Singleton() {
}
public static Singleton getInstance() {
return uniqBean;
}
}
如上代码,利用static的特性,在类加载的同时初始化了Singleton,避免了多线程的问题,只要程序不是严格要求使用时才初始化,这个方法还是不错的。
3.双重检测同步延时加载
public class Singleton {
private volatile static Singleton uniqBean;
private Singleton() {
}
public static Singleton getInstance() {
if (uniqBean == null) {
synchronized (Singleton.class) {
if (uniqBean == null) {
uniqBean = new Singleton();
}
}
}
return uniqBean;
}
}
这个方法的要点1)volatile 要在多个线程之间通信 2)第一个uniqBean == null的判断,这可以在uniqBean不为null时避免同步 3)synchronized 避免两个实例 4)第二个uniqBean == null 假设两个线程同时判断第一个if,一个线程进入synchronized,执行完毕后,uniqBean初始化,另一个线程进入synchronized,所以需要第二次判断啦。
关于volatile,据说jdk1.2之前的版本有问题。关于它实现通信的功能,建议看一下effective java,里面解释很详细。