Bean作用域和生命周期

简介: Bean作用域和生命周期

💚1.Bean 的作用域

🧊1.1Bean的作用域的定义


Bean的作用域和我们之前学过的不一样,我们之前学的作用域是一个范围,而现在指的是 Bean在Spring框架中的某种行为模式,也就是一个动作.

这样干巴巴的说看我可能无法理解,我们来举个例子


🧊1.2代码理解Bean的作用域

创建一个公共类的一个公共对象,两个人各自调用这个方法


package com.java.demo.model;
import org.springframework.context.annotation.Bean;
import org.springframework.stereotype.Component;
import org.springframework.stereotype.Controller;
/**
 * Created with IntelliJ IDEA.
 * Description:
 * User: 小魏
 * Date: 2023-07-29
 * Time: 10:17
 */
/**
 * Users是公共类
 */
@Component
public class Users {
    /**
     * Bean对象是公共对象,默认是单例模式
     * @return
     */
    @Bean("user")
    public User getUser(){
        User user=new User();
        user.setId(1);
        user.setName("lisi");
        return user;
    }
}


用户1


package com.java.demo.controller;
import com.java.demo.model.User;
import com.java.demo.model.Users;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
/**
 * Created with IntelliJ IDEA.
 * Description:
 * User: zh
 * Date: 2023-07-29
 * Time: 10:22
 */
@Controller
public class UserController2 {
    @Autowired
    private User user;
    public void sayHi(){
    User user2=user;
        System.out.println("User"+user2);
    user2.setName("王一博");
        System.out.println("User"+user2);
    }
}


用户2


package com.java.demo.controller;
import com.java.demo.model.User;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
/**
 * Created with IntelliJ IDEA.
 * Description:
 * User: WHY
 * Date: 2023-07-29
 * Time: 10:26
 */
@Controller
public class UserController3 {
    @Autowired
    private User user;
    public void sayhi(){
        System.out.println("User"+user);
    }
}

1f30af9b45c946b7a1c8790cef48d716.png


我们的用户WHY只想单纯的调用公共对象并打印,但是结果变成了用户zh修改以后的代码,这是为啥呢?

因为Bean对象在Spring框架里面就是默认是单例模式的,也就是一份代码里就一个对象,是公共的,一旦有人修改,拿到的数据就是修改以后的


2fe8c06c340941d29b21762a894dc7b3.png


这里定义了一个user2对象指向user对象,所以在后面的修改中,改变了user2也就是改变了user,也就是我们之前学过的浅克隆

那么怎么解决这个问题呢?,采用@Scope注解,Scope就是作用域的意思


bb6887b990f4445fa1f7f4ffbaeb7096.png


prototype代表原型模式,又称多例模式,每次用注解请求bean对象的时候都再次new一个原来的对象

总结一下:Bean的作用域就是Bean在spring容器中的某种行为(比如单例,原型)单例情况下只有一份,修改以后其他人拿到的就是修改后的,加上scope注解,每次都会new一个原来的对象


🧊1.3Bean的六种作用域


🍉1.3.1 singleton:单例作用域


官方说明:(Default) Scopes a single bean definition to a single object instance for each Spring IoC container.

描述:该作⽤域下的Bean在IoC容器中只存在⼀个实例:获取Bean(即通过applicationContext.getBean等⽅法获取)及装配Bean(即通过@Autowired注⼊)都是同⼀个对象。

场景:通常⽆状态的Bean使⽤该作⽤域。⽆状态表示Bean对象的属性状态不需要更新

这个是Spring默认的模式,那么单例模式的Bean是线程安全的吗?

不是,我们可以采用ThreadLocal(本地线程变量)

回忆一下,保证线程安全的方法:

1.使用线程安全的容器2.使用锁.例如synchronized,Lock 3使用ThreadLocal


🍉1.3.2 pprototype:请求作用域

官⽅说明:Scopes a single bean definition to any number of object instances.

描述:每次对该作⽤域下的Bean的请求都会创建新的实例:获取Bean(即通过

applicationContext.getBean等⽅法获取)及装配Bean(即通过@Autowired注⼊)都是新的对象

实例。

场景:通常有状态的Bean使⽤该作⽤域


🍉1.3.3 request:请求作用域

官⽅说明:Scopes a single bean definition to the lifecycle of a single HTTP request. That is,

each HTTP request has its own instance of a bean created off the back of a single bean

definition. Only valid in the context of a web-aware Spring ApplicationContext.

描述:每次http请求会创建新的Bean实例,类似于prototype

场景:⼀次http的请求和响应的共享Bean

限定SpringMVC中使⽤


🍉1.3.4 session:会话作用域

官⽅说明:Scopes a single bean definition to the lifecycle of an HTTP Session. Only valid in

the context of a web-aware Spring ApplicationContext.

描述:在⼀个http session中,定义⼀个Bean实例,一个http共享一个Bean

场景:⽤户回话的共享Bean, ⽐如:记录⼀个⽤户的登陆信息

限定SpringMVC中使⽤


🍉1.3.5application作用域

官⽅说明:Scopes a single bean definition to the lifecycle of a ServletContext. Only valid in the context of a web-aware Spring ApplicationContext.

描述:在⼀个http servlet Context中,定义⼀个Bean实例.一个context容器共享一个作用域

场景:Web应⽤的上下⽂信息,⽐如:记录⼀个应⽤的共享信息

限定SpringMVC中使⽤


🍉1.3.6websocket作用域

官⽅说明:Scopes a single bean definition to the lifecycle of a WebSocket. Only valid in the

context of a web-aware Spring ApplicationContext.

描述:在⼀个HTTP WebSocket的⽣命周期中,定义⼀个Bean实例

场景:WebSocket的每次会话中,保存了⼀个Map结构的头信息,将⽤来包裹客户端消息头。第⼀

次初始化后,直到WebSocket结束都是同⼀个Bean。

限定Spring WebSocket中使⽤


💚2.设置作用域


@Scope 标签既可以修饰⽅法也可以修饰类,@Scope 有两种设置⽅式:

  1. 直接设置值:@Scope(“prototype”)
  2. 使⽤枚举设置:@Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)


🧊单例作⽤域(singleton) VS 全局作⽤域(application)


1.singleton是spring core的作用域,application是Spring web的作用域

2.singleon作用于IOC的容器,application作用于Servlet容器

3.singleton是在一个IOC容器里使用,application是有多个servlet容器里面使用,每个容器都有每个的bean对象


💚3.Spring执行流程和Bean的生命周期

🧊3.1Spring执行流程

1.配置文件加载:Spring启动时,会根据配置文件加载信息,读取XML、注解或Java配置文件中的Bean定义,将其转换为内部的BeanDefinition对象,并放入BeanFactory中。

2.Bean 实例化:当BeanFactory初始化完成后,Spring会根据Bean定义中的配置信息,实例化Bean对象,并将其保存在BeanFactory中。

3.Bean 属性注入:Spring容器会查找所有Bean的属性,查找其是否配置了需要注入的依赖项,如果有,则将依赖项注入到当前Bean中。


4.Bean 初始化:Spring容器会执行定义在Bean上的初始化方法,这些方法可以使用注解或XML配置指定。


5.容器初始化完成:所有Bean初始化完成后,Spring容器会广播一个容器初始化完成事件,通知所有的监听器。


6.应用程序使用:应用程序可以使用Spring容器中的Bean来处理请求或执行任务。


7.Bean销毁:当应用程序关闭时,Spring容器会调用所有Bean的销毁方法,它们可以使用注解或XML配置指定,来清理任何资源,并释放系统资源。


🧊3.2Bean生命周期


拿房子举例🐶

1.实例化(分配内存)[买房,划地]

2.设置Bean属性(DI,将依赖的Bean赋值到当前类的属性上)[需要钢筋水泥工人建房子]

3.Bean的初始化[装修]

3.1执行各种通知,实现了各种 Aware 通知的⽅法,比如BeanNameAwareBeanFactoryAware

3.2初始化的前置方法

3.3@PostConstruct 初始化方法,依赖注入操作之后被执行

3.4执行自己指定的init-method方法(没有就不执行)

3.5初始化的后置方法

4.使用Bean [入住]

5.销毁Bean[房子过期拆房]

执行流程图

c0a31e2160a643baa6f3e2d89fbea2bd.png


说到这里,我们来回忆一下初始化实例化的相关知识

初始化是指创建对象时为对象的属性或变量赋初值。在程序中,通常使用构造函数来完成初始化工作。


实例化是指创建一个类的具体实例(对象)的过程。当类定义完成后,通过使用类来创建对象。


因此,可以说实例化之前需要先进行初始化,而实例化本身是创建对象的过程。


🧊3.3生命周期演示


import org.springframework.beans.factory.BeanNameAware;
import javax.annotation.PostConstruct;
import javax.annotation.PreDestroy;
/**
 * Created with IntelliJ IDEA.
 * Description:
 * User: WHY
 * Date: 2023-07-29
 * Time: 17:49
 */
public class BeanLifeComponent implements BeanNameAware {
    @Override
    public void setBeanName(String s) {
        System.out.println("执行了BeanMNameAware->"+s);//s就是通知的名字,表明给bean起好了名字
    }
    //使用当前注解表示在这个类里面可以实现初始化方法,
    @PostConstruct
    public void doPostConstruct(){
        System.out.println("执行了PostConstruct");
    }
    //使用xml形式也实现了初始化方法
        public void init(){
            System.out.println("执行了init方法");
        }
//销毁bean 1.使用xml形式
    public void destroy(){
        System.out.println("执行了destroy方法");
    }
    //2.使用注解销毁bean
    @PreDestroy
    public void doDestroy(){
        System.out.println("执行了PreDestroy");
    }
    public void sayhi(){
        System.out.println("执行sayhi");
    }
}


这里就是使用xml形式进行注入方法


import org.springframework.context.support.ClassPathXmlApplicationContext;
/**
 * Created with IntelliJ IDEA.
 * Description:
 * User: WHY
 * Date: 2023-07-29
 * Time: 18:09
 */
public class BeanLifeTest {
    public static void main(String[] args) {
        ClassPathXmlApplicationContext context=
                new ClassPathXmlApplicationContext("spring-config.xml");
        BeanLifeComponent beanLifeComponent=context.getBean("mybean",BeanLifeComponent.class);
        beanLifeComponent.sayhi();
        context.close();
    }
}


注意,使用的是ClassPathXmlApplicationContext方法,因为这个子类方法有销毁方法,Application方法没有销毁方法

🧊3.4为什么属性设置要在初始化之前呢?

举个例子

6aa1bf0fdd5447acad82006d7c8cb420.png


这时注入了一个类属性,我有可能会在初始化方法里面调用这个属性的方法,所以当我先设置属性,后面调用的时候就不会报错,如果先初始化后设置属性,一定会报错

今天的讲解就到这里吧,我们下期再见咯

相关文章
|
6月前
|
XML Java 数据格式
SpringBean的生命周期
SpringBean的生命周期
64 0
|
6月前
|
前端开发 Java 开发者
Bean的生命周期和作用域
Bean的生命周期和作用域
|
6月前
|
XML Java 数据格式
Spring框架学习 -- Bean的生命周期和作用域
Spring框架学习 -- Bean的生命周期和作用域
54 2
|
6月前
|
存储 设计模式 Java
Bean 作用域和生命周期
Bean 作用域和生命周期
|
前端开发 Java C++
Bean的作用域
Bean 常见的 6 种作用域
|
Java 容器 Spring
Bean生命周期
Bean生命周期
|
存储 安全 Java
Bean 的作用域和生命周期
Bean 的作用域和生命周期
72 1
|
Java 容器 Spring
bean的作用域和生命周期和后置处理器以及作用域对生命周期的影响~
bean的作用域和生命周期和后置处理器以及作用域对生命周期的影响~
|
XML Java Maven
springBean的作用域
springBean的作用域
70 3
|
Java Spring 容器
Bean的作用域和生命周期(上)
Bean的作用域和生命周期(上)