单例模式作为开发过程中最常用的设计模式之一,是程序员必须了解和掌握的技能,但是单例的实现方式和手段有很多种,每一种都有着各自的优缺点,如何在代码中运用就需要我们对每一种实现方式都了如执掌方可运筹帷幄。
代码
public class Single {
private static Single single = new Single();
// 一定要将默认构造方法设置为private 否则反射实例化将破坏单例
private Single() {
}
public static Single getInstance() {
return single;
}
}
优点
缺点
代码
public class Single {
private static Single single = null;
static{
single = new Single();
}
// 一定要将默认构造方法设置为private 否则反射实例化将破坏单例
private Single() {
}
public static Single getInstance() {
return single;
}
}
优点
缺点
public class Single {
private static Single single = null;
// 一定要将默认构造方法设置为private 否则反射实例化将破坏单例
private Single() {
}
private static Single getInstance() {
// 当多个线程同时执行到这里的时候,会创建出多个对象
if (null == single) {
single = new Single();
}
return single;
}
}
public class Single {
private static Single single = null;
// 一定要将默认构造方法设置为private 否则反射实例化将破坏单例
private Single() {
}
private static synchronized Single getInstance() {
if (null == single) {
single = new Single();
}
return single;
}
}
public class Single {
private static Single single = null;
// 一定要将默认构造方法设置为private 否则反射实例化将破坏单例
private Single() {
}
private static Single getInstance() {
if (null == single) {
// 当多个线程通知执行到这里来了之后,无法保证对象单一
synchronized(Single.class){
single = new Single();
}
}
return single;
}
}
public class Single {
//这里一定要加volatile 否则可能因为指令重排的问题导致对象未初始化完成的情况
private volatile static Single single = null;
// 一定要将默认构造方法设置为private 否则反射实例化将破坏单例
private Single() {
}
public static Single getInstance() {
if (null == single) {
synchronized (Single.class) {
if (null == single) {
single4 = new Single();
}
}
}
return single;
}
}
关于为什么要加volatile,可参考:深度剖析synchronized、volatile的实现细节 其中有详细的说明这个问题。
public class Single {
// 一定要将默认构造方法设置为private 否则反射实例化将破坏单例
private Single() {
}
private static class LazyHolder {
private static final Single INSTANCE = new Single();
}
public static Single getInstance() {
return LazyHolder.INSTANCE;
}
}
public enum Single6
{
INSTANCE;
public void test()
{
System.out.println("test()");
}
}
以上列举了开发过程中所有的单例实现方式,同时还列出了详细的其详细的优缺点;对于枚举方式、DCL方式以及静态内部类的方式是个人相对比较推荐的方式;可以视情况使用;至于饿汉式的两种写法,由于比较简单,如果对应的单例对象在系统中必定用到而且频繁使用,也可以考虑使用;所以,没有更好,只有最好!合适才好!!!