04-Spring中Bean的作用域

Bean的作用域

scope的属性值

属性值 作用
singleton 默认单例
prototype 原型每调用一次getBean()方法则获取一个新的Bean对象 , 每次注入的时候都是新对象
request 一个请求对应一个Bean仅限于在WEB应用中使用 , 需要引入web的框架如SpringMvc
(global) session 一个会话对应一个Bean仅限于在WEB应用中使用 , 需要引入web的框架如SpringMvc
global session portlet应用中专用的, portlet和servlet都是规范, servlet运行在servlet容器中如Tomcat, portlet运行在portlet容器中
application 一个应用对应一个Bean仅限于在WEB应用中使用
websocket 一个websocket生命周期对应一个Bean仅限于在WEB应用中使用
自定义scope 很少使用

scope属性值singleton(单例)

Spring的IoC容器中默认情况下Bean对象是单例的

  • Bean对象的创建是在初始化Spring上下文的时候就完成的 , 执行getBean()方法的时候Bean对象已经创建好了 , 每次获取操作的都是那个单例的Bean对象
java 复制代码
public class SpringBean {
    public SpringBean() {
        System.out.println("SpringBean的无参数构造方法执行");
    }
}
xml 复制代码
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
    <!--默认情况下Bean对象是单例的,scope="singleton"可以省略-->
    <bean id="sb" class="com.powernode.spring6.beans.SpringBean" />
</beans>
java 复制代码
@Test
public void testScope(){
    ApplicationContext applicationContext = new ClassPathXmlApplicationContext("spring-scope.xml");
    SpringBean sb1 = applicationContext.getBean("sb", SpringBean.class);
    //com.powernode.spring6.beans.SpringBean@5b239d7d
    System.out.println(sb1);

    //com.powernode.spring6.beans.SpringBean@5b239d7d
    SpringBean sb2 = applicationContext.getBean("sb", SpringBean.class);
    System.out.println(sb2);
}

scope属性值prototype(多例)

如果想让Spring的Bean对象以多例的形式存在,可以在bean标签中指定scope属性的值为prototype

  • 初始化Spring上下文时不会创建这些prototype的Bean对象 , 只有每一次执行getBean()方法的时候才会创建,调用几次创建几次 , 每次获取的都是新的对象
xml 复制代码
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
    <bean id="sb" class="com.powernode.spring6.beans.SpringBean" scope="prototype" />
</beans>
java 复制代码
@Test
public void testScope(){
    ApplicationContext applicationContext = new ClassPathXmlApplicationContext("spring-scope.xml");
    //com.powernode.spring6.beans.SpringBean@29445a7
    SpringBean sb1 = applicationContext.getBean("sb", SpringBean.class);
    System.out.println(sb1);
	//com.powernode.spring6.beans.SpringBean@67d48005
    SpringBean sb2 = applicationContext.getBean("sb", SpringBean.class);
    System.out.println(sb2);
}

自定义线程级别Scope(了解)

第一步: 自定义Scope实现Scope接口,实现在同一个线程中获取的Bean都是同一个, 跨线程获取的是不同的Bean对象

  • Spring内置了实现Scope接口的线程范围的类org.springframework.context.support.SimpleThreadScope

第二步: 使用CustomScopeConfigurer内置类将自定义的Scope注册到Spring容器中

xml 复制代码
<bean class="org.springframework.beans.factory.config.CustomScopeConfigurer">
  <!--设置scopes属性,这个属性是个map集合表示可以指定多个线程范围-->
  <property name="scopes">
    <map>
      <!--自定义线程名-->
      <entry key="myThread">
        <!--这个Scope接口的实现类使用的是Spring内置的,也可以自己定义-->
        <bean class="org.springframework.context.support.SimpleThreadScope"/>
      </entry>
    </map>
  </property>
</bean>

第三步: 使用自定义线程的Scope

xml 复制代码
<bean id="sb" class="com.powernode.spring6.beans.SpringBean" scope="myThread" />
java 复制代码
@Test
public void testCustomScope(){
    //主线程
    ApplicationContext applicationContext = new ClassPathXmlApplicationContext("spring-scope.xml");
    SpringBean sb1 = applicationContext.getBean("sb", SpringBean.class);
    SpringBean sb2 = applicationContext.getBean("sb", SpringBean.class);
    //com.powernode.spring6.beans.SpringBean@5b236d7d
    System.out.println(sb1);
    //com.powernode.spring6.beans.SpringBean@5b236d7d
    System.out.println(sb2);
    // 启动新线程
    new Thread(new Runnable() {
        @Override
        public void run() {
            SpringBean a = applicationContext.getBean("sb", SpringBean.class);
            SpringBean b = applicationContext.getBean("sb", SpringBean.class);
            //com.powernode.spring6.beans.SpringBean@29445a6
            System.out.println(a);
            //com.powernode.spring6.beans.SpringBean@29445a6
            System.out.println(b);
        }
    }).start();
}
相关推荐
lulu121654407813 分钟前
OpenRouter Fusion 多模型融合架构深度拆解:预算级模型组团打平 Fable 5,多模型协作才是 AGI 的正确打开方式?
java·人工智能·架构·ai编程·agi
雨辰AI18 分钟前
生产级实测:SpringBoot3 + 达梦数据库接口从 200ms 优化至 20ms 完整调优指南
java·数据库·spring boot·后端·政务
Solis38 分钟前
Raft:分布式系统的定海神针
后端·架构
程序员老申38 分钟前
第三篇 5 天 12 个 commit:踩坑实录与代码演进
后端·程序员
程序员鱼皮39 分钟前
提示词工程已死,Loop Engineering 称王!保姆级教程 + 项目实战
前端·后端·ai编程
(Charon)1 小时前
【C++ 面试高频:内存管理、RAII 和智能指针详解】
java·开发语言·word
凡人叶枫1 小时前
Effective C++ 条款39:明智而审慎地使用 private 继承
java·数据库·c++·嵌入式开发
Mininglamp_27181 小时前
Vibe Coding 之后是 Vibe Operating?
后端·开源·多智能体·ai agent·mano-p
星哥的编程之路1 小时前
别再调 API 就说自己会 RAG 了,看看真正的企业级 AI 智能体长什么样
后端·面试
长大19881 小时前
C++26 静态反射完整实战:告别宏代码生成,一键实现序列化
后端