Spring的Bean的作用域的学习

 2023-01-26
原文作者:程序员小徐同学 原文地址:https://juejin.cn/post/7084582900468809735

本文已参与「新人创作礼」活动,一起开启掘金创作之路。

1.bean管理

1.普通bean

在配置文件中定义bean类型就是返回类型

2.工程bean就是FactoryBean

在配置文件定义bean类型可以和返回类型不一样

  • 创建类,让这个类作为工厂bean,实现接口FactoryBean
  • 实现接口里面的方法,在实现的方法中定义返回的bean类型

创建一个不同的返回值

    public class MyBean implements FactoryBean<Course> {
    
    //定义返回Bean
        @Override
        public Course getObject() throws Exception {
            Course course = new Course();
            course.setCname("张思思");
            return course;
        }
    
        @Override
        public Class<?> getObjectType() {
            return null;
        }
    
        @Override
        public boolean isSingleton() {
            return false;
        }
    }
    <?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="myBean" class="com.Spring.Collection.MyBean">
    
    </bean>
    </beans>
    public class Course {
        private String cname;//课程名称
    
        public void setCname(String cname) {
            this.cname = cname;
        }
    
        @Override
        public String toString() {
            return "Course{" +
                    "cname='" + cname + '\'' +
                    '}';
        }
    }

最终是实现Course的类的对象

     @Test
        public void myBean() {
            ApplicationContext context = new ClassPathXmlApplicationContext("bean3.xml");
            Course mybean = context.getBean("myBean", Course.class);
            System.out.println(mybean);
        }

2.Bean的作用域

Spring中,那些组成应用程序的主体及由Spring IoC容器所管理的对象,被称之为bean。简单地讲, bean就是由IoC容器初始化、装配及管理的对象

202301012031188841.png

几种作用域中,request、session作用域仅在基于web的应用中使用(不必关心你所采用的是什么web 应用框架),只能用在基于web的Spring ApplicationContext环境。

Singleton 当一个bean的作用域为Singleton,那么Spring IoC容器中只会存在一个共享的bean实例,并且所有对 bean的请求,只要id与该bean定义相匹配,则只会返回bean的同一实例。Singleton是单例类型,就是 在创建起容器时就同时自动创建了一个bean的对象,不管你是否使用,他都存在了,每次获取到的对象 都是同一个对象。注意,Singleton作用域是Spring中的缺省作用域。要在XML中将bean定义成 singleton,可以这样配置: 1 <bean id="ServiceImpl" class="cn.csdn.service.ServiceImpl" scope="singleton"> 测试

    @Test
    public void test03(){
    ApplicationContext context = new
    ClassPathXmlApplicationContext("applicationContext.xml");
    User user = (User) context.getBean("user");
    User user2 = (User) context.getBean("user");
    System.out.println(user==user2);
    }

Prototype 当一个bean的作用域为Prototype,表示一个bean定义对应多个对象实例。Prototype作用域的bean会 导致在每次对该bean请求(将其注入到另一个bean中,或者以程序的方式调用容器的getBean()方法) 时都会创建一个新的bean实例。Prototype是原型类型,它在我们创建容器的时候并没有实例化,而是 当我们获取bean的时候才会去创建一个对象,而且我们每次获取到的对象都不是同一个对象。根据经 验,对有状态的bean应该使用prototype作用域,而对无状态的bean则应该使用singleton作用域。在 XML中将bean定义成prototype,可以这样配置:

    <bean id="account" class="com.foo.DefaultAccount" scope="prototype"/>
    或者
    <bean id="account" class="com.foo.DefaultAccount" singleton="false"/>

Request 当一个bean的作用域为Request,表示在一次HTTP请求中,一个bean定义对应一个实例;即每个HTTP 请求都会有各自的bean实例,它们依据某个bean定义创建而成。该作用域仅在基于web的Spring ApplicationContext情形下有效。考虑下面bean定义:

    <bean id="loginAction" class=cn.csdn.LoginAction" scope="request"/>

针对每次HTTP请求,Spring容器会根据loginAction bean的定义创建一个全新的LoginAction bean实 例,且该loginAction bean实例仅在当前HTTP request内有效,因此可以根据需要放心的更改所建实例 的内部状态,而其他请求中根据loginAction bean定义创建的实例,将不会看到这些特定于某个请求的 状态变化。当处理请求结束,request作用域的bean实例将被销毁。 Session 当一个bean的作用域为Session,表示在一个HTTP Session中,一个bean定义对应一个实例。该作用域 仅在基于web的Spring ApplicationContext情形下有效。考虑下面bean定义:

    <bean id="userPreferences" class="com.foo.UserPreferences" scope="session"/>

针对某个HTTP Session,Spring容器会根据userPreferences bean定义创建一个全新的 userPreferences bean实例,且该userPreferences bean仅在当前HTTP Session内有效。与request作 用域一样,可以根据需要放心的更改所创建实例的内部状态,而别的HTTP Session中根据 userPreferences创建的实例,将不会看到这些特定于某个HTTP Session的状态变化。当HTTP Session 最终被废弃的时候,在该HTTP Session作用域内的bean也会被废弃掉。

1.设置创建Bean实例是单实例还是多实例

在Spring下默认是单实例对象

1.在Spring配置文件bean标签里面有属性用于设置单实例还是多实例 scope="" 就是设置多实例还是单实例

  • prototype是多实例
  • singleton是单实例也是默认实例
    <?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="book" class="com.Spring.IOC.Book" scope="prototype">
        <property name="bname" value="java开发"></property>
    </bean>
    </beans>
    public class Book {
        private String  bname;
        public void setBname(String bname) {
            this.bname = bname;
        }
    }

202301012031227022.png

prototype与singleton的区别

  • 设置scope是singleton的时候,加载Spring配置文件就会创建单实例对象
  • scope是prototype的时候,不是在加载Spring配置文件时候创建对象,在调用getBean方法时候创建多实例对象

6.Bean的生命周期

1.过程

  • 通过构造器创建Bean实例(无参构造)
  • 为Bean的属性设置值和对其他Bean引用(调用set方法)
  • 调用Bean的初始化方法(需要进行配置初始化方法)
  • Bean可以使用了(对象获取到了)
  • 当容器关闭时,调用Bean的销毁的方法(需要进行配置销毁方法)
    public class Orders {
        private String oname;
    
        public Orders() {
            System.out.println("1.执行无参数构造创建Bean实例");
        }
    
        public void setOname(String oname) {
            this.oname = oname;
            System.out.println("2.调用set方法设置属性值");
        }
        public void initMethod(){
            System.out.println("3.执行初始化的方法");
        }
        public void destroyMethod(){
            System.out.println("5.执行销毁方法");
        }
    }

测试类

     @Test
        public void testBean() {
            ApplicationContext context = new ClassPathXmlApplicationContext("bean3.xml");
            Orders orders = context.getBean("orders2",Orders.class);
            System.out.println("4.获取创建Bean实例对象");
            System.out.println(orders);
            //手动销毁Bean实例
            ((ClassPathXmlApplicationContext) context).close();
        }

xml配置里面要加上init-method 和destory-method这两个方法都在Orders类里面创建

    <bean id="orders2" class="com.Spring.Collection.Orders" init-method="initMethod" destroy-method="destroyMethod">
            <property name="oname" value="手机"></property>
        </bean>

2.添加后置处理器后有七步

  • 通过构造器创建Bean实例(无参构造)
  • 为Bean的属性设置值和对其他Bean引用(调用set方法)
  • 把Bean实例传给Bean后置处理器的方法
      public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
            return bean;
        }
  • 调用Bean的初始化方法(需要进行配置初始化方法)
  • 把Bean实例传递给Bean后置处理器的方法
     public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
            return bean;
        }
  • Bean可以使用了(对象获取到了)
  • 当容器关闭时,调用Bean的销毁的方法(需要进行配置销毁方法)

添加后置处理器方法实现BeanPostProcessor,创建后置处理器

    public class MyBeanPost implements BeanPostProcessor {
        @Override
        public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
            System.out.println("在初始化之前执行的方法");
            return bean;
        }
    
        @Override
        public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
            System.out.println("在初始化方法之后执行的方法");
            return bean;
        }
    }
    public class Orders {
        private String oname;
    
        public Orders() {
            System.out.println("1.执行无参数构造创建Bean实例");
        }
    
        public void setOname(String oname) {
            this.oname = oname;
            System.out.println("2.调用set方法设置属性值");
        }
        public void initMethod(){
            System.out.println("3.执行初始化的方法");
        }
        public void destroyMethod(){
            System.out.println("5.执行销毁方法");
        }
    }

只需要将配置的后置对象加入到配置的文件中,在里面的其他配置实例都会有后置对象

    <?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="orders2" class="com.Spring.Collection.Orders" init-method="initMethod" destroy-method="destroyMethod">
            <property name="oname" value="手机"></property>
        </bean>
        <!--配置后置处理器-->
        <bean id="myBeanPost" class="com.Spring.Collection.MyBeanPost">
    <--7-->
        </bean>
    </beans>

202301012031239023.png