hi,今天为大家带啦Bean的作用域和生命周期的相关知识
Bean的作用域和我们之前学过的不一样,我们之前学的作用域是一个范围,而现在指的是 Bean在Spring框架中的某种行为模式,也就是一个动作.
这样干巴巴的说看我可能无法理解,我们来举个例子
创建一个公共类的一个公共对象,两个人各自调用这个方法
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);
}
}
我们的用户WHY只想单纯的调用公共对象并打印,但是结果变成了用户zh修改以后的代码,这是为啥呢?
因为Bean对象在Spring框架里面就是默认是单例模式的,也就是一份代码里就一个对象,是公共的,一旦有人修改,拿到的数据就是修改以后的
这里定义了一个user2对象指向user对象,所以在后面的修改中,改变了user2也就是改变了user,也就是我们之前学过的浅克隆
那么怎么解决这个问题呢?,采用@Scope注解,Scope就是作用域的意思
prototype代表原型模式,又称多例模式,每次用注解请求bean对象的时候都再次new一个原来的对象
总结一下:Bean的作用域就是Bean在spring容器中的某种行为(比如单例,原型)单例情况下只有一份,修改以后其他人拿到的就是修改后的,加上scope注解,每次都会new一个原来的对象
官方说明:(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
官⽅说明:Scopes a single bean definition to any number of object instances.
描述:每次对该作⽤域下的Bean的请求都会创建新的实例:获取Bean(即通过
applicationContext.getBean等⽅法获取)及装配Bean(即通过@Autowired注⼊)都是新的对象
实例。
场景:通常有状态的Bean使⽤该作⽤域
官⽅说明: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中使⽤
官⽅说明: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中使⽤
官⽅说明: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中使⽤
官⽅说明: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中使⽤
@Scope 标签既可以修饰⽅法也可以修饰类,@Scope 有两种设置⽅式:
1.singleton是spring core的作用域,application是Spring web的作用域
2.singleon作用于IOC的容器,application作用于Servlet容器
3.singleton是在一个IOC容器里使用,application是有多个servlet容器里面使用,每个容器都有每个的bean对象
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配置指定,来清理任何资源,并释放系统资源。
拿房子举例
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[房子过期拆房]
执行流程图
说到这里,我们来回忆一下初始化实例化的相关知识
初始化是指创建对象时为对象的属性或变量赋初值。在程序中,通常使用构造函数来完成初始化工作。
实例化是指创建一个类的具体实例(对象)的过程。当类定义完成后,通过使用类来创建对象。
因此,可以说实例化之前需要先进行初始化,而实例化本身是创建对象的过程。
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");
}
}
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方法没有销毁方法
举个例子
这时注入了一个类属性,我有可能会在初始化方法里面调用这个属性的方法,所以当我先设置属性,后面调用的时候就不会报错,如果先初始化后设置属性,一定会报错