Spring的Bean的作用域的学习

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

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容器初始化、装配及管理的对象

图片.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定义:

扫描二维码关注公众号,回复: 13779383 查看本文章
<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;
    }
}

复制代码

图片.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>
复制代码

图片.png

猜你喜欢

转载自juejin.im/post/7084582900468809735