单例模式

单例模式通俗的讲,有懒汉和饿汉模式,一般很少用饿汉模式,因为考虑到对象的资源消耗等,一般只有在需要的时候才会初始化实例对象,由于多线程实例化,字节码重排序,又有了双重检查锁等实现方式,但由于性能和代码量的因素,内部类和枚举实现是最有效的方式.以下是静态内部类和枚举实现过程.

内部类实现

/**
 * 这些情况下就不用自己再来进行同步控制了。这些情况包括:
 * 
 * 1.由静态初始化器(在静态字段上或static{}块中的初始化器)初始化数据时
 * 
 * 2.访问final字段时
 * 
 * 3.在创建线程之前创建对象时
 * 
 * 4.线程可以看见它将要处理的对象时
 * 
 * @author earthlyfisher
 *
 */
public class SingleInstance {

    private SingleInstance() {
        super();
    }

    /**
     * 类级的内部类,也就是静态的成员式内部类,该内部类的实例与外部类的实例 没有绑定关系,而且只有被调用到时才会装载,从而实现了延迟加载。
     */
    private static class InstanseHolder {
        static{
            System.out.println("内部类加载");
        }
        /**
         * 静态初始化器,由JVM来保证线程安全
         */
        public static final SingleInstance instanse = new SingleInstance();
    }

    public static SingleInstance getInstance() {
        /**
         * 当getInstance方法第一次被调用的时候,它第一次读取SingletonHolder.instance,
         * 导致SingletonHolder类得到初始化;而这个类在装载并被初始化的时候,会初始化它的静态域,从而创建Singleton的实例,
         * 由于是静态的域,因此只会在虚拟机装载类的时候初始化一次,并由虚拟机来保证它的线程安全性。
         */
        return InstanseHolder.instanse;
    }
}

枚举

/**
 * 1.线程安全,通过类加载器保证线程安全。
 * 2.抗序列化问题:枚举类自己实现了readResolve()方法,所以抗序列化,这个方法是当前类自己实现的(解决)
 * @author earthlyfisher
 *
 */
public enum SingleInstanceByEnum {

    instance;

    /**
     * 单例可以有自己的操作
     */
    public void singletonOperation() {
        System.out.println("hello world");
    }
}

你可能感兴趣的:(单例模式)