学无先后,达者为师

网站首页 编程语言 正文

Spring中的单例模式应用详解

作者:搬砖的小熊猫 更新时间: 2024-07-15 编程语言

1. DefaultListableBeanFactory

在Spring中,所有由Spring容器管理的Bean默认都是单例的。Spring框架中最经典的单例模式实现是在BeanFactory中。BeanFactory是Spring IoC容器的核心接口,其实现类DefaultListableBeanFactory在加载Bean定义时,会将单例的Bean实例化并缓存在ConcurrentHashMap中,以保证该Bean的唯一性。

在这里插入图片描述

DefaultListableBeanFactory定义了三个Map对象:singletonObjects、singletonFactories和earlySingletonObjects,它们都被设计为线程安全的ConcurrentHashMap。

  • singletonObjects用于存储已经实例化的单例Bean对象。
  • singletonFactories用于存储BeanFactory对象。
  • earlySingletonObjects用于存储未完全初始化的Bean对象。

当一个单例Bean实例被获取时,DefaultListableBeanFactory会首先检查singletonObjects是否存在该Bean实例,如果存在则直接返回,否则就从earlySingletonObjects或singletonFactories中获取或创建该Bean实例。

2. SingletonBeanRegistry

单例相关的操作其实是被定义在SingletonBeanRegistry接口中。SingletonBeanRegistry是Spring框架中的一个接口,定义了向Spring IoC容器中添加和获取单例Bean的方法。

在这里插入图片描述

public interface SingletonBeanRegistry {

    // 将指定名称的Bean实例注册为单例Bean。如果该名称已经存在于单例Bean注册表中,则会抛出IllegalStateException异常。
    void registerSingleton(String var1, Object var2);

    // 获取指定名称的单例Bean实例。如果指定名称的Bean实例不存在,则返回null。
    @Nullable
    Object getSingleton(String var1);

    // 检查指定名称的单例Bean实例是否已经存在于单例Bean注册表中。
    boolean containsSingleton(String var1);

    // 获取所有已注册的单例Bean名称。
    String[] getSingletonNames();

    // 获取当前容器中已经注册的单例Bean的数量。
    int getSingletonCount();

    // 获取一个用于同步单例Bean注册表的对象。
    Object getSingletonMutex();
}

3. Spring单例Bean与单例模式的区别

Spring单例Bean与单例模式的区别在于它们关联的环境不一样,单例模式是指在一个JVM进程中仅有一个实例,而Spring单例是指一个Spring Bean容器(ApplicationContext)中仅有一个实例。

首先看单例模式,在一个JVM进程中(理论上,一个运行的JAVA程序就必定有自己一个独立的JVM)仅有一个实例,无论在程序中的何处获取实例,始终都返回同一个对象。以Java内置的Runtime为例,下面的判断始终为真:

// 在一个JVM实例中始终只有一个实例
Runtime.getRuntime() == Runtime.getRuntime();

与此相比,Spring的单例Bean是与其容器 ApplicationContext密切相关的,所以在一个JVM进程中,如果有多个Spring容器,即使是单例bean,也一定会创建多个实例,代码示例如下:

public static void main(String[] args) {

    System.out.println(Runtime.getRuntime() == Runtime.getRuntime());

    // 第一个Spring Bean容器
    ClassPathXmlApplicationContext context_1 = new ClassPathXmlApplicationContext("bean.xml");
    Person msb1 = context_1.getBean("person", Person.class);

    // 第二个Spring Bean容器
    ClassPathXmlApplicationContext context_2 = new ClassPathXmlApplicationContext("bean.xml");
    Person msb2 = context_2.getBean("person", Person.class);

    // 这里绝对不会相等,因为创建了多个实例
    System.out.println(msb1 == msb2);
}

以下是Spring的配置文件:

<!-- 即使声明了为单例,只要有多个容器,也一定会创建多个实例 -->
<bean id="person" class="com.mashibing.spring01.demo03.Person" scope="singleton">
    <constructor-arg name="username">
        <value>mashibing</value>
    </constructor-arg>
</bean>

如果不指定bean的类型,Spring框架生成的Bean默认就是单例的(在当前容器里)。

Spring的单例Bean与Spring Bean管理容器密切相关,每个容器都会创建自己独有的实例,所以与GOF设计模式中的单例模式相差极大,但在实际应用中,如果将对象的生命周期完全交给Spring管理(不在其他地方通过new、反射等方式创建),其实也能达到单例模式的效果。

原文链接:https://blog.csdn.net/qq_26893655/article/details/140337231

  • 上一篇:没有了
  • 下一篇:没有了
栏目分类
最近更新