单例模式的六种写法
饿汉式三种:
(饿汉式不存在线程安全问题)
1.直接饿汉式:
public class Singleton{ private Singleton(){};//构造方法私有化 public static final Singleton INSTANCE = new Singleton();//提供一个可访问的静态常量引用 }2.枚举类型的饿汉式:
public enum Singleton{ INSTANCE; //枚举类型:表示该类的对象是有限的几个,我们可以限定为一个,就成了单例。 }3.静态代码块饿汉式:
public class Singleton{ public static final Singleton INSTANCE; private Singleton(){}; static{ INSTANCE = new Singleton(); // .....其他的操作。这种方式适合在初始化阶段还要做一些其他的操作。 } }懒汉式三种:
1.线程不安全:
public class Singleton{ private Singleton(){}; private static Singleton instance; public static getInstance(){ //提供对外可访问的静态工厂方法 if(instance == null){ instance = new Singleton(); } return instance; } }2.双端检测式(线程安全):
public class Singleton { private Singleton() {};//构造函数私有化 private volatile static Singleton instance = null;//用volatile修饰防止指令重排 //静态工厂方法 public static Singleton getInstance() { if(instance == null) { //双重检查机制 synchronized (Singleton.class) { //同步锁 if(instance == null) {//双重检查机制 instance = new Singleton(); } } } return instance; } }3.静态内部类实现(线程安全)
public class Singleton{ private Singleton(){}; //在外部无法访问静态内部类,静态内部类不会随之外部类的加载和初始化而加载初始化,而是在 //调用getInstance方法是才被加载。所以是线程安全的 private static class InnerClass{ private static final Singleton INSTANCE = new Singleton(); } public static Singleton getInstance(){ return InnerClass.INSTANCE; } }利用反射打破单例
//第一步,获得单例类的构造器。 Constructor con = Singleton.class.getDeclaredConstructor(); //第二步,把构造器设置为可访问。 con.setAccessible(true); //第三步,使用newInstance方法构造对象 Singleton singleton1 = (Singleton)con.newInstance(); Singleton singleton2 = (Singleton)con.newInstance(); //验证是否是不同对象 System.out.println(singleton1 == singleton2);