五种单例模式实现

核心作用:

保证一个类只有一个实例,并向外提供一个访问该实例的访问点。

常见场景:

  • 数据库连接池的设计一般也是单例模式
  • 在Servlet编程中,每个Servlet也是单例模式
  • 在Spring中,默认创建的bean也是单例模式
  • 。。。。。。

优点:

1、由于每个类只创建一个实例,大大减少了内存的开销。

2、单例模式提供全局访问点,可以实现共享资源访问。

常见的五种单例模式实现方法:

  • 饿汉式(线程安全,效率高,不能延迟加载)
  • 懒汉式(线程安全,效率不高,可以延迟加载)
  • DCL懒汉式(线程安全,效率还行,可以延迟加载,由于关于JVM底层内部模型原因,即指令重排,偶尔会出现问题,不推荐使用)
  • 静态内部类之饿汉式(线程安全,效率高,可以延迟加载)
  • 枚举单例(线程安全,效率高,不可以延迟加载,用于解决上面4个方法的反射问题)

饿汉式:

public class Singleton01 {
//    私有化构造器
    private Singleton01(){};
//    类加载的时候就创建实例出来
    public static final Singleton01 singleton = new Singleton01();
//      全局访问点没synchronized关键字,所以效率高
    public static Singleton01 getInstance(){
        return singleton;
    }
}

缺点:

如果这个类的实例一直没有被使用,那么也是浪费内存资源。解决这个问题就出现了懒汉式,使用的时候再创建。

懒汉式:

public class Singleton02 {
    private Singleton02(){};
//   先把对象引用设为null
    public static  Singleton02 singleton = null;
//  使用了synchronized关键字,所以效率不高
    public static synchronized Singleton02 getInstance(){
        if(singleton == null){
            singleton = new Singleton02();
        }
        return singleton;
    }
}

缺点:

效率不高。解决这个问题出现了DCL懒汉式,将synchronized锁的代码更加精确

DCL懒汉式:

public class Singleton03 {
    private Singleton03(){};

    public  static  Singleton03 singleton = null;

    public static synchronized Singleton03 getInstance(){
        if(singleton == null){
//            锁住更加精确的代码,尽量让效率提高
            synchronized (Singleton03.class){
                if(singleton == null){
                    singleton = new Singleton03();
                }
            }
        }
        return singleton;
    }
}

缺点:

由于JVM底层内部模型,指令重排,出现问题。具体体现:

由于singleton = new Singleton03()不是原子操作,理论具体操作为:

memory = allocate(); //1:分配对象的内存空间 ctorInstance(memory); //2:初始化对象 instance = memory; //3:设置instance指向刚分配的内存地址

由于指令重排可能变成:

memory = allocate(); //1:分配对象的内存空间 instance = memory; //3:设置instance指向刚分配的内存地址,此时对象还没被初始化 ctorInstance(memory); //2:初始化对象

当进程a执行第二个指令的时候,又有一个进程b访问getInstance方法,这时候singleton不为空,但由于进程a还没初始化,这个进程b就把没初始化的对象返回出去,出现了问题了。

解决方法:

public static Singleton03 singleton = null;改为public volatile static Singleton03 singleton = null;

静态内部类之饿汉式:

public class Singleton04 {
    private Singleton04(){};
    private static class InnerClass{
        private static final Singleton04 singleton = new Singleton04();
    }
    public static Singleton04 getInstance(){
        return InnerClass.singleton;
    }
}

缺点:

反射可以获取类的class对象,修改构造函数的可见性,强制创建一个的对象。

枚举单例:

优点:是为了解决上面四种方法的一个缺点,就是反射问题。

可以产生问题的代码:

Constructor<Singleton04> constructor = Singleton04.class.getConstructor();
constructor.setAccessible(true);
final Singleton04 singleton04 = constructor.newInstance();

解决方法:

通过查看Constructor的源码:

@CallerSensitive
@ForceInline // to ensure Reflection.getCallerClass optimization
public T newInstance(Object ... initargs)
    throws InstantiationException, IllegalAccessException,
           IllegalArgumentException, InvocationTargetException
{
    if (!override) {
        Class<?> caller = Reflection.getCallerClass();
        checkAccess(caller, clazz, clazz, modifiers);
    }
    if ((clazz.getModifiers() & Modifier.ENUM) != 0)
        throw new IllegalArgumentException("Cannot reflectively create enum objects");
    ConstructorAccessor ca = constructorAccessor;   // read volatile
    if (ca == null) {
        ca = acquireConstructorAccessor();
    }
    @SuppressWarnings("unchecked")
    T inst = (T) ca.newInstance(initargs);
    return inst;
}

可以看到当类的为枚举类型,即使改了可见性,当执行newInstance也会抛出IllegalArgumentException错误出来,不能实例化一个对象。

所以将单例模式的类设置为枚举类,代码为:

public enum Singleton05 {
    SINGLETON;

    public Singleton05 getSingleton(){
        return SINGLETON;
    }
}

缺点:

不能延迟加载。

原文地址:https://www.cnblogs.com/zitai/p/12219756.html

时间: 2024-10-13 00:38:35

五种单例模式实现的相关文章

五种单例模式

今天算是完完整整把老高的单例模式看了,差不多也懂了,个别不懂的再补吧,下面介绍一下5种单例模式: 饿汉式实现:线程安全,调用效率高.但是,不能延时加载. 懒汉式实现:线程安全,调用效率不高.但是,可以延时加载. 双重检测锁式:由于JVM底层内部模型原因,偶尔会出现问题,不建议使用. 静态内部类式:线程安全,调用效率高,可以延时加载. 枚举式:线程安全,调用效率高.但是,不能延时加载(避免反射和反序列化的漏洞). 以下先介绍一下5种单例模式: 饿汉式实现: /** *单例模式:饿汉模式 *线程安全

Java中的五种单例模式实现方法

[代码] Java中的五种单例模式实现方法 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 package s

五种单例模式----来自脚本之家

本文为大家分享了Python创建单例模式的5种常用方法,供大家参考,具体内容如下 所谓单例,是指一个类的实例从始至终只能被创建一次. 方法1: 如果想使得某个类从始至终最多只有一个实例,使用__new__方法会很简单.Python中类是通过__new__来创建实例的: ? 1 2 3 4 5 6 7 8 9 10 11 12 13 class Singleton(object):   def __new__(cls,*args,**kwargs):     if not hasattr(cls,

快速理解Java中的五种单例模式

解法一:只适合单线程环境(不好) package test; /** * @author xiaoping * */ public class Singleton { private static Singleton instance=null; private Singleton(){ } public static Singleton getInstance(){ if(instance==null){ instance=new Singleton(); } return instance;

常见的五种单例模式实现方式

--主要: 饿汉式(线程安全,调用效率高,但是不能延时加载) 懒汉式(线程安全,调用效率低,但是可以延时加载) --其他: 双重检测锁式(由于JVM底层内部模型原因,偶尔会出现问题,不建议使用) 静态内部类式(线程安全,调用效率高,可以延时加载) 枚举式(线程安全,调用效率高,不能延时加载) -- 如何选用? 单例对象   占用   资源 少,不需要   延时加载:  枚举式   好于  饿汉式 单例对象   占用   资源 大, 需要   延时加载:   静态内部类式   好于   懒汉式 引

C# 单例模式的五种写法

C# 单例模式的五种写法及优劣分析,见下文: [单例模式及常见写法](http://blog.csdn.net/jiankunking/article/details/50867050)

单例模式的五种写法

首先来明确一个问题,那就是在某些情况下,有些对象,我们只需要一个就可以了,比如,一台计算机上可以连好几个打印机,但是这个计算机上的打印程序只能有一个,这里就可以通过单例模式来避免两个打印作业同时输出到打印机中,即在整个的打印过程中我只有一个打印程序的实例. 简单说来,单例模式(也叫单件模式)的作用就是保证在整个应用程序的生命周期中, 任何一个时刻,单例类的实例都只存在一个(当然也可以不存在). package singleton; /** * @author lei * 单例模式的五种写法: *

五种方法实现Java的Singleton单例模式

面试的时候经常会问到Java的单例模式,这道题能很好的考察候选人对知识点的理解程度.单例模式要求在系统运行时,只存在唯一的一个实例对象. 下面我们来详细剖析一下其中的关键知识点,并介绍五种实现方法,以及它们的优缺点. 一.最简单的方法是在类加载的时候初始化这个单独的实例. 首先,定义单例类(没有特别的,就是起个名字): 1 public class Singleton{ 其次,需要定义类变量将单例对象保存下来: 1 private static Singleton instance = new

php 常用五种模式

/* 设计模式之单例模式 $_instance 必须声明为静态的私有变量 构造函数必须声明为私有,防止外部程序 new 类从而失去单例模式的意义 getInstance() 方法必须设置为公有的,必须调用此方法以返回实例的一个引用 :: 操作符只能访问静态变量和函数 new 对象会消耗内存 使用场景:最常用的地方是数据库连接 使用单例模式生成一个对象后,该对象可以被其它众多对象所使用 [单例模式适用场景] 1.当类只能有一个实例而且客户可以从一个众所周知的访问点访问它时 2.当这个唯一实例应该是