美文网首页
单例模式(Singleton)

单例模式(Singleton)

作者: CallMe兵哥 | 来源:发表于2020-01-17 14:20 被阅读0次

    单例模式(Singleton)

    ​ Singleton模式,主要是一般为了保证自己研发的系统或者软件中只有一个实例,如果使用 Java 的理解就是只有一个对象。因为对象是可以通过 new 的方式来创建,所以为了保证我们的系统里面只有一个对象,必须要做一些约定,然后这些约定和实现,可以称之为单例模式。

    ​ 在这里,我说个我年轻粗浅理解的一个笑话,我看到单例模式?以为无论如何使用都是单例的,所以我在构思一个软件的时候,想着在另外一个Java应用中,直接调用另外一个Java应用中的单例对象,试图操作另外一个应用。经过几次尝试,我发现不管我怎么设置,另外一个应用就是对我的设置不做任何响应,那时候我觉得委屈极了,怎么就不对呢?现在想想真是一个愚蠢的笑话,当时就没弄明白应用(也可理解为Windows里面的进程)之间是隔离的、独立的,如果想获得另外一个应用的对象,必须通过通信或者文件共享。我说这个笑话主要是想告诉你:假如你也闹了一些类似的笑话,没关系,因为,很多人也闹过。如果你不想再闹这些笑话,只能学习。

    ​ 单例模式的实现有很多种方式,下面我简单介绍一些常用的写法。

    饿汉式

    ​ 这个模式的好处就是简单、方便,也便于理解,唯一实例是依靠 JVM 来实现。主要的开发源码如下:

    /**
     * 饿汉式
     * 类加载到内存以后,直接new一个单例,由JVM保证线程的安全
     * 简单实用
     *
     * 问题:无论用到与否,类都需要装载。
     */
    public class Mgr01 {
        private static final Mgr01 INSTANCE = new Mgr01();
    
        private Mgr01() {}
    
        public static Mgr01 getInstance() {
            return INSTANCE;
        }
    
        public void out() {
            System.out.println("this is Mgr01");
        }
    
        public static void main(String[] args) {
            for (int i = 0; i < 20; i++) {
                new Thread(()->{
                    Mgr03 mgr03 = Mgr03.getInstance();
                    //我们输出线程的hash值,假如是同一个对象,hash值肯定一样。
                    System.out.println(mgr03.hashCode());
                }).start();
            }
        }
    }
    

    懒汉式

    ​ 从上面的用例我们可以明显的看到,饿汉式的核心问题,与java的核心 LazyLoading(懒加载)理念不一样,众所周知 Java 是懒加载的,跑性能前,一般都建议先跑30分钟预热,因为java是越跑越快(这个原理,请翻阅JVM相关资料)。所以有人说第一个方式不合理。又提出了LazyLoading 的方案。

    /**
     * lazy loading
     * 懒汉式加载
     * 虽然达到了懒加载的目的,但是带来了线程不安全的问题
     */
    public class Mgr03 {
        private static Mgr03 INSTANCE;
    
        private Mgr03() {}
    
        public static Mgr03 getInstance(){
            if (null == INSTANCE) {
                //假如这里阻塞了,或者
                try {
                    Thread.sleep(1);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                INSTANCE = new Mgr03();
    
            }
            return INSTANCE;
        }
    
        public void out () {
            System.out.println("this is Mgr03");
        }
    
        public static void main(String[] args) {
    
            for (int i = 0; i < 20; i++) {
                new Thread(()->{
                    Mgr03 mgr03 = Mgr03.getInstance();
                    //我们输出线程的hash值,假如是同一个对象,hash值肯定一样。
                    System.out.println(mgr03.hashCode());
                }).start();
            }
        }
    }
    

    双重判空加锁

    ​ 前面写的 LazyLoading 加载方案,明显存在并发问题,如果并发上来以后,对象可能就不是同一个,所以提出了加锁 synchronized 的方式。如果把 synchronized 加到方法上,固然能解决问题,但是肯定会带来性能问题。所以专家提议把锁加到判断空以后的方法块上面,所以有了双重判空的方式。就有如下代码。

    /**
     * lazy loading
     * 懒汉式加载
     * 虽然达到了懒加载的目的,但是带来了线程不安全的问题
     *
     * 双重判空加锁
     */
    public class Mgr06 {
        private static Mgr06 INSTANCE;
    
        private Mgr06() {}
    
        public static Mgr06 getInstance(){
            if (null == INSTANCE) {
                synchronized (Mgr06.class){
                    if (null == INSTANCE) {
                        //假如这里阻塞了,或者
                        try {
                            Thread.sleep(2100);
                        } catch (InterruptedException e) {
                            e.printStackTrace();
                        }
                        INSTANCE = new Mgr06();
                    }
                }
            }
            return INSTANCE;
        }
    
        public void out () {
            System.out.println("this is Mgr03");
        }
    
        public static void main(String[] args) {
            System.out.println("Thread start ..........");
            for (int i = 0; i < 20; i++) {
                new Thread(()->{
                    Mgr06 mgr03 = Mgr06.getInstance();
                    //我们输出线程的hash值,假如是同一个对象,hash值肯定一样。
                    System.out.println(mgr03.hashCode());
                }).start();
            }
        }
    }
    

    静态内部类

    ​ 这个方式,其实之前我是没接触过的,看完这种方式以后,觉得这个操作真是骚,比双重判空感觉优雅多了。这个方式,其实是利用了Java静态内部类(也是依靠JVM来保证)。因为JVM在加载Java类时,如果没有用到是不会创建静态的内部类。这样又达到了 LazyLoading 的目的,又避免了加锁。加锁一般情况下是不建议加锁的,加锁会浪费系统资源。

    /**
     * lazy loading
     * 懒汉式加载
     *
     * 静态内部类的方法
     * JVM保证单例
     * 加载外部类时,不会加载到内部类,这样可以实现懒加载
     */
    public class Mgr07 {
    
        private Mgr07() {}
    
        private static class Mgr07Holder {
            private final static Mgr07 INSTANCE = new Mgr07();
        }
    
        public static Mgr07 getInstance(){
            //假如这里阻塞了,或者
            try {
                Thread.sleep(2100);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            return Mgr07Holder.INSTANCE;
        }
    
        public void out () {
            System.out.println("this is Mgr03");
        }
    
        public static void main(String[] args) {
            System.out.println("Thread start ..........");
            for (int i = 0; i < 20; i++) {
                new Thread(()->{
                    Mgr07 mgr07 = Mgr07.getInstance();
                    //我们输出线程的hash值,假如是同一个对象,hash值肯定一样。
                    System.out.println(mgr07.hashCode());
                }).start();
            }
        }
    }
    

    枚举单例

    ​ 面对网络上面的众多关于单例的讨论和方式,让我们的Java创始人之一的某位大神也产生了兴趣,最后终极大招,枚举单例。这种模式可以防止反序列号,大哥出手,一看就有。具体的写法如下。

    /**
     *
     * java创始人之一 ,写个了更完美的,这种写法,可以防止反序列化
     * 枚举单例
     */
    public enum Mgr08 {
    
        INSTANCE;
    
        public void out () {
            System.out.println("this is Mgr03");
        }
    
        public static void main(String[] args) {
            System.out.println("Thread start ..........");
            for (int i = 0; i < 20; i++) {
                new Thread(()->{
                    //我们输出线程的hash值,假如是同一个对象,hash值肯定一样。
                    System.out.println(Mgr08.INSTANCE.hashCode());
                }).start();
            }
        }
    }
    

    ​ 最后,单例模式使用范围非常广,几乎所有的开源产品都使用了单例模式,例如一个数据库连接池、一个线程池、读取配置文件等等,Spring 的 IOC 容器里面,默认的对象,都是单例的。所以单例这种设计模式,一定要充分的理解和掌握。

    相关文章

      网友评论

          本文标题:单例模式(Singleton)

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