【Spring Boot】Spring原理:Bean的作用域和生命周期

Spring原理

一. 知识回顾

在之前IOC/DI的学习中我们也用到了Bean对象,现在先来回顾一下IOC/DI的知识吧!
首先Spring IOC,也叫控制反转,简单来说就是依赖添加5大注解把该对象交给Spring来管理,Spring会把该对象放入IOC容器中,在接下来的调用中直接注入即可,注入也就是Spring DI操作了。
回顾一下,一共有以下五大注解:

1.1 回顾Spring IOC

类注解

  1. @Controller(控制层注解)
  2. @Service(逻辑层注解)
  3. @Repository(数据层注解)
  4. @Component(总注解)
  5. @Configuration(插件注解)

然后就是有一点要注意的就是,@Component注解可以说是其他四个注解的父注解,就是其他注解底层都是依赖@Component来实现的,都可以使用@Component注解来代替使用,但是不能代替@Controller,因为别忘了Controller注解还有返回视图的作用,这是@Component注解所不具备的

方法注解

  1. @Bean(方法注解)

该注解用于把方法交给Spring进行管理,但是必须和类注解连用

1.2 回顾Spring DI

Spring DI就是把IOC容器里的东西拿出来进行使用,主要是@Autowired注解,主要有三种注入方式

  1. 属性注入(就是通过给成员变量进行注入)
  2. 构造注入(就是通过构造方法注入)
  3. Set方法注入

其实在实际运用中,使用属性注入基本上能满足90%的需求了.

Spring DI主要面试考的主要是拥有多个相同对象,注入时该如何保证?

主要提供了以下三大注解

  1. @Primary(默认注入的方法)
  2. @Qualifier(加入要注入对象的方法名称)
  3. @Resource(要注入对象的名称)

@Autowird与@Resource的区别

  1. @Autowired是spring框架提供的注解,⽽@Resource是JDK提供的注解
  2. @Autowired默认是按照类型注⼊,⽽@Resource是按照名称注⼊.相⽐于@Autowired来说,@Resource⽀持更多的参数设置,例如name设置,根据名称获取Bean。

1.3 回顾如何获取对象

Spring主要提供了两种方法获取Bean对象:

  1. ApplicationContext(上下文)
  2. @Autowired

第二种是注入方式就不一一叙述了,主要是ApplicationContext获取Bean对象,其实就是调用了分类BeanFactory工厂来获取对象。
两者主要有以下两点区别:

  1. 继承关系和功能方⾯来说:Spring容器有两个顶级的接口:BeanFactory和
    ApplicationContext。其中BeanFactory提供了基础的访问容器的能⼒,⽽
    ApplicationContext属于BeanFactory的⼦类,它除了继承了BeanFactory的所有功能之外,
    它还拥有独特的特性,还添加了对国际化⽀持、资源访问⽀持、以及事件传播等方⾯的⽀持.

  2. 从性能方⾯来说:ApplicationContext是⼀次性加载并初始化所有的Bean对象,也就是饿加载,⽽
    BeanFactory是需要那个才去加载那个,也就是懒加载,因此更加轻量.(空间换时间)

好了,回顾完了Spring IOC/DI的知识点了,就该进入正题了。

二. Bean的作用域

Bean的作用域是指Bean在Spring框架中的某种行为模式.

主要有以下6种作用域

  1. singleton:单例作用域
  2. prototype:原型作用域(多例作用域)
  3. request:请求作用域
  4. session:会话作用域
  5. Application:全局作用域
  6. websocket:HTTPWebSocket作用域
作用域 说明
singleton 每个SpringIoC容器内同名称的bean只有⼀个实例(单例)(默认)
prototype 每次使用该bean时会创建新的实例(⾮单例)
request 每个HTTP请求生命周期内,创建新的实例(web环境中)
session 每个HTTPSession生命周期内,创建新的实例(web环境中)
application 每个ServletContext生命周期内,创建新的实例(web环境中)
websocket 每个WebSocket生命周期内,创建新的实例(web环境中)

单例作用域:多次访问,得到的都是同⼀个对象,并且 @Autowired 和 applicationContext.getBean()
也是同⼀个对象.

多例作用域:观察ContextDog,每次获取的对象都不⼀样(注⼊的对象在Spring容器启动时,就已经注⼊了,所以多次请求也不会发生变化)

请求作用域:在⼀次请求中, @Autowired 和 applicationContext.getBean() 也是同⼀个对象.
但是每次请求,都会重新创建对象

会话作用域:在⼀个session中,多次请求,获取到的对象都是同⼀个,换⼀个浏览器访问,发现会重新创建对象.(另⼀个Session)

Application作用域:在⼀个应用中,多次访问都是同⼀个对象

注意:Applicationscope就是对于整个web容器来说,bean的作用域是ServletContext级别的.这个和
singleton有点类似,区别在于:Applicationscope是ServletContext的单例,singleton是⼀个
ApplicationContext的单例.在⼀个web容器中ApplicationContext可以有多个

三. Bean的生命周期

生命周期指的是⼀个对象从诞生到销毁的整个生命过程,我们把这个过程就叫做⼀个对象的生命周期.
Bean的生命周期分为以下5个部分:

  1. 实例化(为Bean分配内存空间)

  2. 属性赋值(Bean注⼊和装配,⽐如 @AutoWired )

  3. 初始化
    执行各种通知,如 BeanNameAware , BeanFactoryAware ,ApplicationContextAware 的接口方法.

  4. 使用Bean

  5. 销毁Bean
    销毁容器的各种方法,如 @PreDestroy , DisposableBean 接口方法, destroymethod.

在这里插入图片描述
实现的代码如下:

@Component
public class BeanLifeComponent implements BeanNameAware {

	private UserComponent userComponent;
	
	public BeanLifeComponent() {
		System.out.println("执行构造函数");
	}
	
	@Autowired
	public void setUserComponent(UserComponent userComponent) {
		System.out.println("设置属性userComponent");
		this.userComponent = userComponent;
	}
	
	@Override
	public void setBeanName(String s) {
		System.out.println("执行了 setBeanName 方法:" + s);
	}
	
	/**
	* 初始化
	*/
	@PostConstruct
	public void postConstruct() {
		System.out.println("执行 PostConstruct()");
	}
	
	public void use() {
		System.out.println("执行了use方法");
	}
	
	/**
	* 销毁前执行方法
	*/
	@PreDestroy
	public void preDestroy() {
		System.out.println("执行:preDestroy()");
	}
}

相关推荐

  1. 深入理解Spring框架中Bean作用生命周期

    2024-07-11 21:48:03       58 阅读

最近更新

  1. docker php8.1+nginx base 镜像 dockerfile 配置

    2024-07-11 21:48:03       67 阅读
  2. Could not load dynamic library ‘cudart64_100.dll‘

    2024-07-11 21:48:03       71 阅读
  3. 在Django里面运行非项目文件

    2024-07-11 21:48:03       58 阅读
  4. Python语言-面向对象

    2024-07-11 21:48:03       69 阅读

热门阅读

  1. 机器学习深度学习用得到的数据集

    2024-07-11 21:48:03       29 阅读
  2. 设计模式:单例模式

    2024-07-11 21:48:03       22 阅读
  3. BP神经网络与反向传播算法在深度学习中的应用

    2024-07-11 21:48:03       22 阅读
  4. Ubuntu 打开或关闭界面

    2024-07-11 21:48:03       18 阅读
  5. c++状态机实现示例

    2024-07-11 21:48:03       23 阅读
  6. Mac上pyenv的安装及使用

    2024-07-11 21:48:03       18 阅读
  7. 【C++】智能指针

    2024-07-11 21:48:03       24 阅读
  8. 【TS】Typescript 的类

    2024-07-11 21:48:03       23 阅读
  9. 说一下浏览器中的强缓存和协商缓存的区别

    2024-07-11 21:48:03       24 阅读
  10. 【Redis 如何实现分级缓存】

    2024-07-11 21:48:03       20 阅读