欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

互联网架构-精讲设计模式-010:深入理解单例模式

程序员文章站 2022-06-15 11:40:50
...

1 饿汉式模式

课程内容:
1、什么是单例模式?单例的应用场景
2、完全解密单例的七种写法
3、如何去**一个单例,创建多次
4、如何防止反射、序列化**单例

什么是单例模式
单例模式确保一个类只有一个实例,并提供一个全局访问点,实现单例模式的方法是私有化构造函数,通过getInstance()方法实例化对象,并返回这个实例。

单例模式优缺点
优点
1、单例类只有一个实例
2、共享资源,全局使用
3、节省创建时间,提高性能
缺点:可能存在线程不安全的问题

单例的七种写法
分别是「饿汉」、「懒汉(非线程安全)」、「懒汉(线程安全)」、「双重校验锁」、「静态内部类」、「枚举」和「容器类管理」

饿汉式

public class SingletonV1 {

    /**
     * 饿汉式 优点:先天线程安全 当类被加载的时候就会创建该对象
     * 缺点:如果项目中使用过多饿汉式,项目在启动的时候非常慢,存放在方法区占用内存比较大
     * 如果用户不使用该对象的时候,也会被提前创建,浪费资源
     */
    private static SingletonV1 singletonV1 = new SingletonV1();

    private SingletonV1() {
    }

    /**
     * 返回该对象的实例
     *
     * @return
     */
    public static SingletonV1 getInstance() {
        return singletonV1;
    }
}
public class SingletonV1Test {
    public static void main(String[] args) {
        SingletonV1 instance1 = SingletonV1.getInstance();
        SingletonV1 instance2 = SingletonV1.getInstance();
        System.out.println(instance1 == instance2); // true
    }
}

优点:先天性线程是安全的,当类初始化的时候就会创建该对象
缺点:如果饿汉式使用过多,可能会影响项目启动的效率问题。

2 懒汉式模式(线程不安全)

懒汉式(线程不安全)

public class SingletonV2 {

    // 懒汉式 当真正需要使用该对象的时候才会被初始化
    private static SingletonV2 singletonV2;

    private SingletonV2(){}

    /**
     * 可能存在线程安全问题,在多线程的情况下,可能会被初始化多次
     * @return
     */
    public static SingletonV2 getInstance(){
        // 当第一次singletonV2等于null的情况下才会被初始化,第二次直接返回对象
        if(singletonV2 == null){
            singletonV2 = new SingletonV2();
        }
        return singletonV2;
    }
}
public class SingletonV2Test {
    public static void main(String[] args) {
        SingletonV2 instance1 = SingletonV2.getInstance();
        SingletonV2 instance2 = SingletonV2.getInstance();
        System.out.println(instance1 == instance2); // true
    }
}

懒汉式模拟高并发场景线程不安全问题

public class SingletonV2 {

    // 懒汉式 当真正需要使用该对象的时候才会被初始化
    private static SingletonV2 singletonV2;

    private SingletonV2(){}

    /**
     * 可能存在线程安全问题,在多线程的情况下,可能会被初始化多次
     * @return
     */
    public static SingletonV2 getInstance(){
        // 当第一次singletonV2等于null的情况下才会被初始化,第二次直接返回对象
        if(singletonV2 == null){
            try{
                Thread.sleep(3000);
            }catch(Exception e){
                e.printStackTrace();
            }
            singletonV2 = new SingletonV2();
        }
        return singletonV2;
    }
}
public class SingletonV2Test {
    public static void main(String[] args) {
        // 模拟高并发情况下 懒汉式线程安全问题
        for (int i = 0; i < 300; i++) {
            new Thread(new Runnable() {
                @Override
                public void run() {
                    SingletonV2 singletonV2 = SingletonV2.getInstance();
                    System.out.println(Thread.currentThread().getName() + "," + singletonV2);
                }
            }).start();
        }
    }
}

互联网架构-精讲设计模式-010:深入理解单例模式
在getInstance()方法上加上synchronized可以解决线程安全问题,但是运行如同单线程,效率较低。

3 双重检验锁原理

懒汉式解决线程安全问题为什么效率比较低?
线程安全问题,当多个线程共享同一个数据做写的操作,可能会存在线程安全问题,读不存在线程安全问题。
懒汉式解决线程安全(加上synchronized)读和写都加上了锁,应该是第一次创建对象的时候才会加锁,之后获取该对象的时候不需要加锁。

双重检验锁目的是什么?
解决懒汉式(线程安全)获取对象效率问题。

双重检验锁

public class SingletonV3 {

    /**
     * volatile 防止重排序 java内存模型 保证可见性
     */
    private volatile static SingletonV3 singletonV3;
    // 双重检验锁 解决懒汉式读和写都加锁
    private SingletonV3(){}

    public static SingletonV3 getInstance() {

        // 当多个线程同时可能在new对象的时候,才会加锁,保证线程安全问题
        if (singletonV3 == null) {
            synchronized (SingletonV3.class) {
                if (singletonV3 == null) { // 当前线程已经获取到锁了,再判断该对象是否已经初始化,没有初始化的话开始创建
                    singletonV3 = new SingletonV3();
                }
            }
        }
        return singletonV3;
    }

}

注:如果没有第二个if (singletonV3 == null),假设第一个if (singletonV3 == null)同时有10个线程进入,synchronized只是保证一个线程获取到锁,第一个获取锁的线程最先创建对象,后面9个线程随后获取锁同样也会创建对象,导致一共创建10个对象。而如果有第二个if,那只有第一个获取锁的线程创建对象成功,后面线程不创建,保证单例。

测试效果:

public class SingletonV3 {

    /**
     * volatile 防止重排序 java内存模型 保证可见性
     */
    private volatile static SingletonV3 singletonV3;
    // 双重检验锁 解决懒汉式读和写都加锁
    private SingletonV3(){}

    public static SingletonV3 getInstance() {

        // 当多个线程同时可能在new对象的时候,才会加锁,保证线程安全问题
        if (singletonV3 == null) {
            try{
                Thread.sleep(3000);
            }catch(Exception e){
                e.printStackTrace();
            }
            synchronized (SingletonV3.class) {
                if (singletonV3 == null) { // 当前线程已经获取到锁了,再判断该对象是否已经初始化,没有初始化的话开始创建
                    singletonV3 = new SingletonV3();
                }
            }
        }
        return singletonV3;
    }

}
public class SingletonV3Test {
    public static void main(String[] args) {
        // 模拟高并发情况下 懒汉式线程安全问题
        for (int i = 0; i < 300; i++) {
            new Thread(new Runnable() {
                @Override
                public void run() {
                    SingletonV3 singletonV3 = SingletonV3.getInstance();
                    System.out.println(Thread.currentThread().getName() + "," + singletonV2);
                }
            }).start();
        }
    }
}

互联网架构-精讲设计模式-010:深入理解单例模式第一次执行(创建对象)慢,因为走了sleep方法,后面执行都快。

4 静态内部类方式

如何解决写和读都不加锁,还能保证唯一性,线程安全问题?
静态内部类

public class SingletonV4 {

    private SingletonV4() {
        System.out.println("构造函数被初始化...");
    }

    public static SingletonV4 getInstance() {
        return SingletonV4Utils.singletonV4;
    }

    // 在类里面嵌套的
    private static class SingletonV4Utils {
        private static final SingletonV4 singletonV4 = new SingletonV4();
    }

    /**
     * 内部类在调用的时候才会初始化singletonV4
     * static 静态 保证唯一
     * @param args
     */
    // 静态内部类特征:继承懒汉式和饿汉式优点、同时解决双重检验锁第一次加载慢的问题 读和写都不需要同步效率非常高...
    public static void main(String[] args) {
        System.out.println("项目启动成功...");
        SingletonV4 instance1 = SingletonV4.getInstance();
        SingletonV4 instance2 = SingletonV4.getInstance();
        System.out.println(instance1 == instance2); // true
    }
}

懒加载:内部类在调用的时候才会创建对象
保证线程安全问题:静态对象static保证唯一性

互联网架构-精讲设计模式-010:深入理解单例模式

5 使用反射技术**单例

单例基本原则:保证在单个jvm中不重复创建
虽然单例通过私有构造函数,可以实现防止程序员初始化对象,但是还可以通过反射和序列化技术**单例。

使用反射技术**单例

public class SingletonV3Test2 {
    public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
        SingletonV3 instance1 = SingletonV3.getInstance();
        // 如何去**单例 使用java反射技术、序列化技术
        Constructor<SingletonV3> declaredConstructor = SingletonV3.class.getDeclaredConstructor();
        declaredConstructor.setAccessible(true);
        // 使用java的反射技术创建
        SingletonV3 instance2 = declaredConstructor.newInstance();
        System.out.println(instance1 == instance2); //false
    }
}

如何防止被反射**----私有化构造函数

private SingletonV3() throws Exception {
        if(singletonV3 !=null){
            throw new Exception("对象已经被初始化...");
        }
        System.out.println("SingletonV3被初始化...");
    }

互联网架构-精讲设计模式-010:深入理解单例模式

6 使用序列化**单例

反序列化创建对象**单例

public class SingletonV5 implements Serializable {
    public static void main(String[] args) throws IOException, ClassNotFoundException {
        // java的序列化技术
        /**
         * 对象从内存写入到硬盘中 序列化
         * 从硬盘中读取到内存 反序列化
         */
        SingletonV5 instance1 = SingletonV5.getInstance();
        FileOutputStream fos = new FileOutputStream("D:\\code\\Singleton.obj");
        ObjectOutputStream oos = new ObjectOutputStream(fos);
        oos.writeObject(instance1);
        oos.flush();
        oos.close();

        // 反序列化
        FileInputStream fis = new FileInputStream("D:\\code\\Singleton.obj");
        ObjectInputStream ois = new ObjectInputStream(fis);
        SingletonV5 instance2 = (SingletonV5) ois.readObject();
        System.out.println(instance1 == instance2); //false
    }

    private static SingletonV5 singletonV5 = new SingletonV5();

    private SingletonV5() {}

    public static SingletonV5 getInstance() {
        return singletonV5;
    }

/*    //返回序列化获取对象 ,保证为单例
    public Object readResolve() {
        return singletonV5;
    }*/

}

互联网架构-精讲设计模式-010:深入理解单例模式
互联网架构-精讲设计模式-010:深入理解单例模式

7 枚举方式防止**

枚举方式防止**

public enum EnumSingleton {
    INSTANCE;

    // 枚举能够绝对有效的防止实例化多次,和防止反射和序列化**
    public void add() {
        System.out.println("add方法...");
    }
}
public class EnumSingletonTest {
    public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
        EnumSingleton instance1 = EnumSingleton.INSTANCE;
        EnumSingleton instance2 = EnumSingleton.INSTANCE;
        System.out.println(instance1 == instance2);
        instance1.add();

        Constructor<EnumSingleton> declaredConstructor = EnumSingleton.class.getDeclaredConstructor();
        declaredConstructor.setAccessible(true);
        EnumSingleton instance3 = declaredConstructor.newInstance();
        System.out.println(instance3 == instance1);
    }
}

互联网架构-精讲设计模式-010:深入理解单例模式
枚举类没有无参构造函数,不能通过反射方式创建对象。