当前位置: 首页>编程语言>正文

springboot 非单例bean spring中bean是单例吗


文章目录

  • Bean的作用域概述
  • 格式(以singleton为例)
  • 详细说说
  • Singleton
  • Prototype
  • Request
  • Session
  • Global Session
  • 生命周期
  • Bean实例化过程
  • Bean生命周期


spring中,哪些构成应用程序的主体以及被Spring IOC容器托管的对象叫做Bean

Bean的作用域概述

在Spring中,Bean默认都是单例的
在Java中,单例是基于JVM的,每个JVM中只有一个实例。
而Spring中的单例是基于BeanFactory的,也就是Spring IOC容器,Spring IOC的底层实质是一个工厂模式
Bean有五种作用域,其中三种仅在基于Web的应用中使用(无论使用的Web框架是什么),只能用在基于Web的SpringApplicationContext环境

下面简单的介绍一下五种作用域

类别

简介

singleton

在Spring IOC容器中仅存在一个Bean实例,Bean以单例方式存在,也是默认作用域

prototype

每次从容器中调用Bean时,都返回一个新的实例,即每次调用getBean()时,都相当于执行new XXX()

request

每次HTTP请求都会创建一个新的Bean,该作用域仅适用于WebApplicationContext环境

session

同一个Http Session共享一个Bean,不同Session使用不同Bean,仅适用于WebApplicationContext

globalSession

一般用于Portlet应用环境,该作用域仅适用于WebApplicationContext环境

格式(以singleton为例)

xml格式

<bean class="com.dean.controller" id="userController" scope="singleton"></bean>

注释格式

@Componet("userController")
@Scope(value="singleton")
//加在Controller类上

详细说说

Singleton

  • 当一个Bean的作用域为Singleton,那么Spring IOC容器中只会存在一个共享的bean实例,并且当所有对bean的请求,只要id与该bean定义相匹配,则只会返回bean的同一实例
  • Singleton是单例类型,即创建容器时就同时自动创建了一个Bean对象,不论是否被使用,它都已经存在了,每次获取到的对象也都是同一对象。
  • 可以对非线程安全的对象采用单实例模式
  • 虽然启动时,自动实例化bean会花费一些时间,但它却带来了两个好处:
    对bean提前的实例化操作,会及早发现一些潜在的配置问题
    Bean以缓存的方式运行,当运行到需要使用该bean的时候,就不需要再去实例化了,加快了运行效率

Prototype

  • 一个Bean定义对应多个对象实例
  • Prototype的作用域会导致在每次对该bean请求时,都会创建一个新的Bean实例,
  • 在创建容器时并没有实例化,当我们去获取Bean实例的时候才会去创建一个对象,而且每次获取到的对象都不是同一个对象
  • 有一点非常重要,Spring不能对一个prototype bean的整个生命周期负责:容器在初始化、装配或装配完一个prototype实例后,将它交给客户端,随后就对该prototype实例不闻不问了
  • 清除prototype作用域的对象并释放任何prototype bean所持有的昂贵资源,都是客户端代码的职责

Request

  • 在一次HTTP请求中,一个Bean定义对应一个实例
  • 每个Http请求都会有各自的bean实例
  • 针对每次HTTP请求,Spring容器会根据loginAction bean的定义创建一个全新的LoginAction bean实例,该LoginAction bean实例仅在当前Http request内有效,因此可以根据需要放心的更改所建实例的内部状态,而其他请求中根据loginAction bean定义所创建的实例,将不会看到这些特定于某个请求的状态变化

Session

  • 在一个Http Session中,一个Bean定义对应一个实例,作用域仅在Web的Spring ApplicationContext情形下有效
  • 针对某个Http Session,Spring容器会根据userPerferences bean定义创建一个全新的userPerferences bean实例,且该userPerferences bean仅在当前Http Session内有效
  • 根据需要放心的更改所创建实例的内部状态,而别的Http Session中根据userPreferences创建的实例,将不会看到这些特定于某个HTTP Session的状态变化

Global Session

  • 在一个全局Http Session中,一个bean定义对应一个实例。
  • 仅在基于portlet的web应用中才有意义Portlet规范定义了全局Session的概念

生命周期

以下图来自网络(图方便理解以下文字)

Bean实例化过程

springboot 非单例bean spring中bean是单例吗,springboot 非单例bean spring中bean是单例吗_作用域,第1张

  • Spring对bean进行实例化,默认bean的作用域是单例
  • Spring对bean进行依赖注入
  • 如果bean实现了BeanNameAware接口,Spring将bean的id传给setBeanName()方法
  • 如果bean实现了BeanFactoryAware接口,Spring调用setBeanFactory方法,将BeanFactory实例传进来
  • 如果bean实现了ApplicationContextAware接口,它的setApplicationContext()方法将被调用,将应用上下文的引用传入到bean中
  • 如果bean实现了BeanPostProcessor接口,它的postProcessBeforeInitialization方法将被调用
  • 如果bean实现了InitializingBean接口,spring将调用它的afterPropertiesSet接口方法,类似的如果bean使用了init-method属性声明了初始化方法,该方法也会调用
  • 如果bean实现BeanPostProcessor接口,它的postProcessAfterInitialization接口方法将被调用
  • 此时bean以及准备就绪,可以被应用程序使用,他们将一直驻留在应用上下文中,直到该应用上下文被销毁
  • 若bean实现了DisposableBean接口,spring将调用它的distroy()接口方法。同样的,如果bean使用了destroy-method属性声明了销毁方法,则该方法被调用
  • 实际上:很多时候并不会去实现上面所描述的哪些接口

Bean生命周期

  • init-method 指定一个方法,在实例化bean时,立即调用该方法
  • destroy-method 指定一个方法,只有从容器中移除bean之后,才能调用该方法
  • default-init-method和default-destory-method 在你有太多同名初始化和销毁方法的bean时,不需要在每一个bean上声明初始化和销毁方法,而是用上述两个设置默认值



https://www.xamrdz.com/lan/5qu1938514.html

相关文章: