• 单例模式


    定义: 一个类只能产生一个实例,并且该类提供一个唯一的全局访问点。

    说明: 在设计系统时,有的对象,需要保证全局只有一个。例如如下对象(线程池,log对象, 注册表信息对象,性能设置对象,驱动类对象,打印机对象等 )

    基本思路:

      1. 私有构造器。保证其他类无法实例化该类。

      2. 静态方法。通过该静态方法能得到该对象的实例。

    Java中有以下几种写法:

    一. 懒汉(线程不安全)

    public class Singleton {
    
        private static Singleton instance;
    
        private Singleton() {
    
        }
    
        public static Singleton getInstance() {
            if (instance == null) {
                instance = new Singleton();
            }
            return instance;
        }
    }

    总结: 这种写法起到了lazy loading的效果,即先判断实例是否存在,不存在的情况,再实例化。但是只能在单线程的情况下使用。在多线程的情况下,假如A线程进入到if(instance == null), 但是B线程已经执行到 instance = new Singleton(), 这种情况下,就会产生产生多个实例,未达到单例的效果。

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

    public class Singleton {
    
        private static Singleton instance;
    
        private Singleton() {
    
        }
    
        public static synchronized Singleton getInstance() {
            if (instance == null) {
                instance = new Singleton();
            }
            return instance;
        }
    }

    总结:通过在方法前加关键字 synchronized来起到线程同步的作用,解决线程不安全。具备lazy loading效果。

    缺点:效率太低。每个线程在执行getInstance() 方法时,都会进行同步。实际上,该方法仅仅需要在第一次执行时需要同步,99%情况下不需要同步,所以影响性能。

    三. 懒汉(同步代码块)

    public class Singleton {
        private static Singleton instance;
    
        private Singleton() {
    
        }
    
        public static Singleton getInstance() {
            if (instance == null) {
                synchronized (Singleton.class) {
                    instance = new Singleton();
                }
            }
            return instance;
        }
    }

    总结: 同步了代码块,但是仔细分析,多线程下,也会出现 写法一的情况,有可能会产生多个实例。

    四. 饿汉(静态变量)

    public class Singleton {
        private final static Singleton instance = new Singleton();
    
        private Singleton() {
    
        }
    
        public static Singleton getInstance() {
            return instance;
        }
    }

    优点:写法简单,在类装载阶段就完成实例化(单例模式中,大多数是调用getInstance()方法时类装载)。基于classloader机制,避免线程同步问题

    缺点:没有lazy loading效果。如果一直没有使用这个实例,会造成内存的浪费。

    五. 饿汉(静态代码块)

    public class Singleton {
        private static Singleton instance;
    
        static {
            instance = new Singleton();
        }
    
        public static Singleton getInstance() {
            return instance;
        }
    }

    总结: 和上面的,也就是第三种效果差不多。只是把实例化过程放在了静态代码块中,也是在类装载的时候,执行静态代码块,初始化类的实例。

    六. 双重校验锁

    public class Singleton {
        private static volatile Singleton instance;
    
        private Singleton() {
    
        }
    
        public static Singleton getInstance() {
            if (instance == null) {
                synchronized (Singleton.class) {
                    if (instance == null) {
                        instance = new Singleton();
                    }
                }
            }
            return instance;
        }
    }

    总结:因为进行了两次if(instance == null)的检查,保证了线程的安全性。

    优点:具有lazy loading效果;线程安全;效率高。

    缺点:使用了volatile 关键字,JDK1.5以后才支持。

     七. 静态内部类

    public class Singleton {
    
        private static class SingletonKeeper {
            private static final Singleton instance = new Singleton();
        }
    
        private Singleton() {
    
        }
    
        public static Singleton getInstance() {
            return SingletonKeeper.instance;
        }
    }

    总结: 利用classloader的机制来保证初始化instance实例时只有一个线程。和写法四 和 写法五,还是有区别的(只要Singleton类被装载,那么instance就会被实例化,也就没有lazy loading效果)。 而此写法,当Singleton类被装载了,instance不一定被实例化,因为SingletonKeeper类没有被主动使用,只有显示通过调用getInstance方法时,才会显示装载SingletonKeeper类,从而实例化instance。

    类的静态属性只会在第一次加载类的时候初始化,所以在这里,JVM帮助我们保证了线程的安全性,在类进行初始化时,别的线程是无法进入的。

    优点:避免了线程不安全,延迟加载,效率高。

    八. 枚举

    public enum Singleton {
        INSTANCE;
        public void whateverMethod() {
    
        }
    }

    总结: 借助JDK1.5中添加的枚举来实现单例模式。不仅能避免多线程同步问题,而且还能防止反序列化重新创建新的对象。

    =====================================================================

  • 相关阅读:
    CF1137FMatches Are Not a Child‘s Play【LCT】
    P4491[HAOI2018]染色【多项式,二项式反演】
    P3170[CQOI2015]标识设计【插头dp】
    log4j 使用教程说明
    log4j中Logger.getLogger()加载一个类提示错误
    编程基础 0x00008 的0x代表什么?
    编程基础 快速的进行 2进制,10进制,16进制 的 相互转换
    Java 基础 equals,hashcode和==的区别
    位运算 左移右移运算符 >>, <<, >>>
    Java 虚拟机 2.2 运行时数据区 Runtime Data Area
  • 原文地址:https://www.cnblogs.com/FocusIN/p/6618811.html
Copyright © 2020-2023  润新知