美文网首页
(10) 单例设计模式(1)

(10) 单例设计模式(1)

作者: 21号新秀_邓肯 | 来源:发表于2021-04-24 18:29 被阅读0次

    1. 基本介绍

    所谓类的单例设计模式,就是采取一定的方法保证在整个的软件系统中,对某个类只能存在一个对象实例,并且该类只提供一个取得其对象实例的方法(静态方法)。

    比如 Hibernate 的 SessionFactory,它充当数据存储源的代理,并负责创建 Session 对象。SessionFactory 并不是轻量级的,一般情况下,一个项目通常只需要一个 SessionFactory 就够,这是就会使用到单例模式。

    2. 单例设计模式八种方式

    • \color{ red} {1) 饿汉式(静态常量)}

    • \color{ red} {2) 饿汉式(静态代码块)}

      1. 懒汉式(线程不安全)
      1. 懒汉式(线程安全,同步方法)
      1. 懒汉式(线程安全,同步代码块)
    • \color{ red} {6) 双重检查}

    • \color{ red} {7) 静态内部类}

    • \color{ red} {8) 枚举}

    3. 饿汉式(静态常量)

    3.1 步骤

    1.构造器私有化 (防止 new )

    2.类的内部创建对象

    3.向外暴露一个静态的公共方法。getInstance

    3.2 代码实现

    1. 饿汉式(静态变量)

    class Singleton {
        
        //1. 构造器私有化, 外部能new
        private Singleton() {
            
        }
        
        //2.本类内部创建对象实例
        private final static Singleton instance = new Singleton();
        
        //3. 提供一个公有的静态方法,返回实例对象
        public static Singleton getInstance() {
            return instance;
        }
        
    }
    

    2. 测试

        public static void main(String[] args) {
            //测试
            Singleton instance = Singleton.getInstance();
            Singleton instance2 = Singleton.getInstance();
            System.out.println(instance == instance2); // true
            System.out.println("instance.hashCode=" + instance.hashCode());
            System.out.println("instance2.hashCode=" + instance2.hashCode());
        }
    

    3.3 优缺点说明

    1. 优点:这种写法比较简单,就是在类装载的时候就完成实例化。避免了线程同步问题。

    2. 缺点:在类装载的时候就完成实例化,没有达到 Lazy Loading 的效果。如果从始至终从未使用过这个实例,则会造成内存的浪费

    3. 这种方式基于 classloder 机制避免了多线程的同步问题,不过,instance 在类装载时就实例化,在单例模式中大多数都是调用 getInstance 方法, 但是导致类装载的原因有很多种,因此不能确定有其他的方式(或者其他的静态方法)导致类装载,这时候初始化 instance 就没有达到 lazy loading 的效果

    4. 结论:这种单例模式可用,可能造成内存浪费

    4. 饿汉式(静态代码块)

    4.1 代码实现

    1.饿汉式(静态变量)

    class Singleton {
        
        //1. 构造器私有化, 外部能new
        private Singleton() {
            
        }
        
        //2.本类内部创建对象实例
        private  static Singleton instance;
        
        static { // 在静态代码块中,创建单例对象
            instance = new Singleton();
        }
        
        //3. 提供一个公有的静态方法,返回实例对象
        public static Singleton getInstance() {
            return instance;
        }
        
    }
    

    2. 测试

        public static void main(String[] args) {
            //测试
            Singleton instance = Singleton.getInstance();
            Singleton instance2 = Singleton.getInstance();
            System.out.println(instance == instance2); // true
            System.out.println("instance.hashCode=" + instance.hashCode());
            System.out.println("instance2.hashCode=" + instance2.hashCode());
        }
    

    4.2 优缺点说明

    1. 这种方式和上面的方式其实类似,只不过将类实例化的过程放在了静态代码块中,也是在类装载的时候,就执行静态代码块中的代码,初始化类的实例。优缺点和上面是一样的。

    2. 结论:这种单例模式可用,但是可能造成内存浪费

    5. 懒汉式(线程不安全)

    5.1 代码实现

    1. 懒汉式

    class Singleton {
        private static Singleton instance;
        
        private Singleton() {}
        
        //提供一个静态的公有方法,当使用到该方法时,才去创建 instance
        //即懒汉式
        public static Singleton getInstance() {
            if(instance == null) {
                instance = new Singleton();
            }
            return instance;
        }
    }
    

    2. 测试

        public static void main(String[] args) {
            System.out.println("懒汉式1 , 线程不安全~");
            Singleton instance = Singleton.getInstance();
            Singleton instance2 = Singleton.getInstance();
            System.out.println(instance == instance2); // true
            System.out.println("instance.hashCode=" + instance.hashCode());
            System.out.println("instance2.hashCode=" + instance2.hashCode());
        }
    

    5.2 优缺点说明

    1. 起到了\color{ red} {Lazy Loading} 的效果,但是只能在单线程下使用。

    2. 如果在多线程下,一个线程进入了 if (singleton == null)判断语句块,还未来得及往下执行,另一个线程也通过了这个判断语句,这时便会产生多个实例。所以在多线程环境下不可使用这种方式

    3. 结论:\color{ red} {在实际开发中,不要使用这种方式.}

    6. 懒汉式(线程安全,同步方法)

    6.1 代码实现

    1. 懒汉式(线程安全,同步方法)

    class Singleton {
        private static Singleton instance;
        
        private Singleton() {}
        
        //提供一个静态的公有方法,加入同步处理的代码,解决线程安全问题
        //即懒汉式
        public static synchronized Singleton getInstance() {
            if(instance == null) {
                instance = new Singleton();
            }
            return instance;
        }
    }
    

    2 测试

        public static void main(String[] args) {
            System.out.println("懒汉式2 , 线程安全~");
            Singleton instance = Singleton.getInstance();
            Singleton instance2 = Singleton.getInstance();
            System.out.println(instance == instance2); // true
            System.out.println("instance.hashCode=" + instance.hashCode());
            System.out.println("instance2.hashCode=" + instance2.hashCode());
        }
    

    6.2 优缺点说明

    1. 解决了\color{ red} {线程安全}问题

    2. 效率太低了,每个线程在想获得类的实例时候,执行 getInstance()方法都要进行同步。而其实这个方法只执行一次实例化代码就够了,后面的想获得该类实例,直接 return 就行了。方法进行同步效率太低

    3. 结论:在实际开发中,\color{ red} {不推荐}使用这种方式

    相关文章

      网友评论

          本文标题:(10) 单例设计模式(1)

          本文链接:https://www.haomeiwen.com/subject/tkmcrltx.html