【Spring】Bean的作用域和生命周期

目录

一、引入案例来探讨Bean的作用域

二、Bean的作用域

2.1、Bean的6种作用域

2.2、设置Bean的作用域

三、Spring的执行流程

 四、Bean的声明周期

1、生命周期演示


一、引入案例来探讨Bean的作用域

首先我们创建一个User类,定义一个用户信息,在定义一个Users类,使用方法注解将user存入Spring中,然后两个用户A对这个公共的Bean获取到之后,在自己的类中对Bean进行了修改,我们预期的结果是公共的Bean可以在各自的类中被修改,但是不能影响到其他类。

package com.java.demo.model;public class User {private int id;private String name;public int getId() {return id;}public void setId(int id) {this.id = id;}public String getName() {return name;}public void setName(String name) {this.name = name;}@Overridepublic String toString() {return "User{" +"id=" + id +", name='" + name + '\'' +'}';}}
package com.java.demo.model;import org.springframework.context.annotation.Bean;
import org.springframework.stereotype.Component;
/*
* 公共对象
* */@Component
public class Users {@Bean("user")public User getUser(){User user = new User();user.setId(123);user.setName("李逵");return user;}
}

 A想着在自己的类中对Bean对象进行修改,不会影响到其他的类。

package com.java.demo.controller;import com.java.demo.model.User;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;/*
* 用户控制器
* 作者:A
* */
@Controller
public class UserController2 {@Autowiredprivate User user;public void doMethod(){User user2 = user;System.out.println("User -> 修改之前 : user "+user);user2.setId(111);user2.setName("黑旋风");System.out.println("User -> 修改之后 : user "+user);}
}

 B没有做任何修改,只是在自己的类中获取了之后,打印了这个Bean对象

package com.java.demo.controller;import com.java.demo.model.User;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;/*
* 用户控制器
* 作者:B
* */
@Controller
public class UserController3 {@Autowiredprivate User user;public void doMethod(){System.out.println("UserController3: user->"+user);}
}

 可以看到A用户在自己的类中修改了Bean对象,也影响到了B获取的Bean对象的结果。这个问题的原因在于Bean对象在默认情况下是单例模式,也就是A和B用户使用的都是同一个对象。为了达到我们预期的效果,下面我们来了解一下Bean的作用域。 


二、Bean的作用域

我们在学习Java基础知识的时候,了解到的作用域是说源代码中定义的变量的可用范围。但是在Spring中Bean的作用域试着Bean在Spring整个框架中的某种行为模式,就比如我们刚刚的代码,它其中的Bean对象的作用域就是单例作用域。

2.1、Bean的6种作用域

Spring容器在初始化一个Bean的实例时,同时会指定该实例的作用域。

1️⃣singleton(单例作用域):是Spring中,Bean默认的作用域,若一个Bean的作用域是单例的,那么每个IoC容器只会有一个Bean对象,所有对这个Bean的依赖和获取这个Bean的代码,拿到的都是同一个Bean对象,这个Bean对象是全局共享的。其他人修改了这个值之后,那么另一个人读取到的就是被修改的值。单例模式的Bean不是线程安全的,可以将Bean里面的属性设置为ThreadLocal(本地线程),就会是线程安全的

需要注意的是这里的单例和我们之前说到的设计模式中的单例是不同的。设计模式中说到的单例模式指的是某个类在进程中只有唯一的一个实例;而Spring中的单例指的是在一个Spring容器中,只会缓存某个类的一个Bean对象,所有通过这个容器获取Bean的方式,拿到的都是同一个Bean对象。但是在不同的Spring容器中,每一个Spring容器都会存在某个类的唯一的一个Bean对象。也就是说这里的单例是限定在一个Spring容器中,而不是整个应用程序中。

2️⃣prototype(原型作用域)也可以理解为多例作用域。若一个Bean的作用域是prototype,那么Spring容器并不会缓存创建的Bean,程序中对这个Bean的每一次获取,容器都会重新实例化一个Bean对象。这也就意味着容器不会帮我们做对象销毁的工作。

3️⃣request(请求作用域)它将Bean的使用范围限定在一个http请求中,对于每个请求,都会单独创建一个Bean,一次的请求和响应共享一个Bean。请求结束,Bean也会随之销毁,使用request作用域一般不存在线程安全问题。

4️⃣session(会话作用域):它将Bean的使用范围限定在了一次http会话中,对于每一个会话,Spring容器都会创建一个单独的Bean,若session被销毁,则Bean也随之销毁。

5️⃣application(全局作用域):在整个Web应用期间,创建一个Bean实例。适合存储全局的配置数据等。

6️⃣websocket(HTTP WebSocket 作用域):在⼀个HTTP WebSocket的⽣命周期中,定义⼀个Bean实例。WebSocket 的每次会话中,保存了⼀个 Map 结构的头信息,将⽤来包裹客户端消息头。第⼀次初始化后,直到 WebSocket 结束都是同⼀个Bean。


2.2、设置Bean的作用域

我们可以使用注解@Scope来声明Bean的作用域,@Scope注解既可以修饰方法也可以修饰类,它的声明的方式有两种。

  1. 直接设置值:@Scope("prototype")
  2. 使用枚举设置:@Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE) 

 下面我们将开始的案例设置有单例作用域修改为原型作用域。

package com.java.demo.model;import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Scope;
import org.springframework.stereotype.Component;
/*
* 公共对象
* */@Component
public class Users {@Bean("user")@Scope("prototype")public User getUser(){User user = new User();user.setId(123);user.setName("李逵");return user;}
}


三、Spring的执行流程

Bean 执⾏流程(Spring 大体执⾏流程):启动 Spring 容器 -> 实例化 Bean(分配内存空间,从⽆到 有) -> Bean 注册到 Spring 中(存操作) -> 将 Bean 装配到需要的类中(取操作)。

 

 四、Bean的声明周期

 生命周期指的是一个对象从创建到销毁的整个生命过程,所以这里的Bean的生命周期指的就是Bean从创建到使用再到销毁的过程。

1️⃣实例化Bean:给Bean分配内存空间,(相当于毛坯房

2️⃣设置属性:当前类创建Bean对象时,依赖其他的Bean对象,这个时候使用(属性注入、Setter注入,构造方法注入)的方式引入依赖的Bean对象,赋值给当前类的属性。(相当于购买装修的基本材料)

3️⃣Bean的初始化:这里相当于装修房子

  • 执行各种通知:相当于通知各个装修师傅来施工
  • 初始化的前置方法:相当于师傅到达现场之后,和业主商量装修的方案
  • 初始化方法:这里初始化的方式有两种一种是使用xml的方式,一种使用注解@PostConstruct的方式;相当于师傅开始经行装修的工作
  • 初始化的后置方法:相当于房子装修完毕之后的清理工作。

4️⃣使用Bean:相当于房子可以入住了

5️⃣销毁Bean:相当于拆或者买了房子

❓❓❓为什么设置属性在初始化之前进行?


❗❗❗通过下面的代码来理解,BeanLifeComponent类中有一个Users类型的变量。如果BeanLifeComponent类中的方法调用users对象的方法时,users对象还没有被引入(也就是说还没有引入这个依赖),那么执行时程序一定会报错。所以就需要先引入这个依赖(Bean对象)。先执行属性设置,有助于当前类创建对象,因为当前类中的方法可能使用了这个属性,属性就必须先赋值完成。

public class BeanLifeComponent implements BeanNameAware {@Autowiredprivate Users users;@Overridepublic void setBeanName(String s) {//通知方法System.out.println("执行了 BeanNameAware ->"+s);}@PostConstructpublic void doPostConstruct(){//使用注解的初始化方式System.out.println("执行了@PostConstruct");}
}

1、生命周期演示

下面我们通过这个例子来说明

import org.springframework.beans.factory.BeanNameAware;import javax.annotation.PostConstruct;
import javax.annotation.PreDestroy;public class BeanLifeComponent implements BeanNameAware {@Overridepublic void setBeanName(String s) {//通知方法System.out.println("执行了 BeanNameAware ->"+s);}@PostConstructpublic void doPostConstruct(){//使用注解的初始化方式System.out.println("执行了@PostConstruct");}public void myInit(){//使用xml的初始化方式System.out.println("执行了myInit");}public void sayHi(){System.out.println("Bean");}@PreDestroypublic void preDestroy(){//销毁方法System.out.println("执行了preDestroy方法");}
}
import org.springframework.context.support.ClassPathXmlApplicationContext;public class BeanLifeTest {public static void main(String[] args) {ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("spring-config.xml");BeanLifeComponent component = context.getBean("myBean", BeanLifeComponent.class);component.sayHi();context.close();//这里会调用preDestroy方法进行对象销毁}
}

配置信息中Bean的init-method属性表示初始化方法

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xmlns:content="http://www.springframework.org/schema/context"xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd"><content:component-scan base-package="com.java"></content:component-scan><bean id="myBean" class="BeanLifeComponent" init-method="myInit"></bean></beans>

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/80936.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

fabric.js里toDataURL后,画布内容展示不全?

复现场景&#xff1a; 用fabric生成画布后&#xff0c;转成图片&#xff0c;然后直接在浏览器里打开&#xff0c;画布展示内容缺失 画布原图&#xff1a; toDataURL后链接在浏览器打开&#xff1a; 原因解析&#xff1a; base64链接太长&#xff0c;输入浏览器链接被截断&…

尚品汇总结九:RabbitMQ在项目的应用(面试专用)

项目中的问题 1.搜索与商品服务的问题 商品服务修改了 商品的上架状态,商品就可以被搜索到.采用消息通知,商品服务修改完商品上架状态,发送消息 给 搜索服务,搜索服务消费消息,进行商品数据ES保存.下架也是一样. 2.订单服务取消订单问题 延迟队里 保存订单之后 开始计时,…

参考RabbitMQ实现一个消息队列

文章目录 前言小小消息管家1.项目介绍2. 需求分析2.1 API2.2 消息应答2.3 网络通信协议设计 3. 开发环境4. 项目结构介绍4.1 配置信息 5. 项目演示 前言 消息队列的本质就是阻塞队列&#xff0c;它的最大用途就是用来实现生产者消费者模型&#xff0c;从而实现解耦合以及削峰填…

如何将视频转成gif图?视频怎么转gif高清图片?

在看电视或是短视频的时候&#xff0c;总能发现一些有趣的片段&#xff0c;当想把这些视频转gif图片发送给朋友的时候该怎么处理呢&#xff1f;其实可以试试专业的视频转gif工具&#xff0c;本文介绍一个视频在线转gif的方法&#xff0c;一起来了解一下吧。 打开首页&#xff…

重发布选路问题

一、思路 &#xff1b; 1.增加不优选路开销解决选路不佳问题 2.用增加开销的方式使R1 不将ASBR传的R7传给另一台ASBR解决R1、R2、R3、R4pingR7环回环路 二、操作 ------IP地址配置如图 1.解决环路 [r2] ip ip-prefix a permit 7.7.7.0 24 [r2]route-policy huawei per…

c++ boost circular_buffer

boost库中的 circular_buffer顾名思义是一个循环缓冲器&#xff0c;其 capcity是固定的当容量满了以后&#xff0c;插入一个元素时&#xff0c;会在容器的开头或结尾处删除一个元素。 circular_buffer为了效率考虑&#xff0c;使用了连续内存块保存元素 使用固定内存&#x…

The Sandbox 与 D.OASIS 联手打造 D.OASIS 城市

我们非常高兴地宣布与 D.OASIS 建立合作伙伴关系&#xff0c;共同打造无与伦比的娱乐体验&#xff1a;The Sandbox 中的 D.OASIS 城市&#xff01; 作为合作的一部分&#xff0c;The Sandbox 和D.OASIS将共同打造 D.OASIS 城市&#xff0c;一座充满无限可能的大都市&#xff0…

CSS基础介绍笔记1

官方文档 CSS指的是层叠样式&#xff08;Cascading Style Sheets&#xff09;地址&#xff1a;CSS 教程离线文档&#xff1a;放大放小&#xff1a;ctrl鼠标滚动为什么需要css&#xff1a;简化修改HTML元素的样式&#xff1b;将html页面的内容与样式分离提高web开发的工作效率&…

Grafana 曲线图报错“parse_exception: Encountered...”

问题现象 配置的Grafana图报错如下&#xff1a; 原因分析 点开报错&#xff0c;可以看到报错详细信息&#xff0c;是查询语句的语法出现了异常。 变量pool的取值为None 解决方案 需要修改变量pool的查询SQL&#xff0c;修改效果如下&#xff1a; 修改后&#x…

华为OD机试(含B卷)真题2023 算法分类版,58道20个算法分类,如果距离机考时间不多了,就看这个吧,稳稳的

目录 一、数据结构1、线性表2、优先队列3、滑动窗口4、二叉树5、并查集6、栈 二、算法1、基础算法2、字符串3、图4、动态规划5、数学 三、漫画算法2&#xff1a;小灰的算法进阶参与方式 很多小伙伴问我&#xff0c;华为OD机试算法题太多了&#xff0c;知识点繁杂&#xff0c;如…

机器学习中的工作流机制

机器学习中的工作流机制 在项目开发的时候&#xff0c;经常需要我们选择使用哪一种模型。同样的数据&#xff0c;可能决策树效果不错&#xff0c;朴素贝叶斯也不错&#xff0c;SVM也挺好。有没有一种方法能够让我们用一份数据&#xff0c;同时训练多个模型&#xff0c;并用某种…

Java源码-Context源码解析

您好&#xff0c;我们来一起了解一下Java源码中的Context源码解析。 Context是Android中的一个重要的概念&#xff0c;在Android开发中可以用来获取应用程序的各种信息&#xff0c;如Activity、Service、Application等等。在Android中&#xff0c;Context是一个抽象类&#xf…

Apache+Tomcat 整合

目录 方式一&#xff1a;JK 1、下载安装包 2、添加依赖 3、启动服务&#xff0c;检查端口是否监听 4、提供apxs命令 5、检查是否确实依赖 6、编译安装 7、重要配置文件 方式二&#xff1a;http_proxy 方式三&#xff1a;ajp_proxy 方式一&#xff1a;JK 1、下载安装…

【大数据】Flink 详解(二):核心篇 Ⅰ

Flink 详解&#xff08;二&#xff09;&#xff1a;核心篇 Ⅰ 14、Flink 的四大基石是什么&#xff1f; ​ Flink 的四大基石分别是&#xff1a; Checkpoint&#xff08;检查点&#xff09;State&#xff08;状态&#xff09;Time&#xff08;时间&#xff09;Window&#xff…

Java、Android 之 TCP / IP

TCP、IP是一系列协议组成的网络分层模型 客户端向服务端发送请求可能会走N条链路&#xff0c;这个过程叫路由 TCP传输 一般在1--1024端口 必须连接以后才能传输数据 UDP协议通常只是发送数据 TCP连接 TCP需要建立连接才能通信&#xff0c;建立连接需要端口&#xff0c;Sock…

20.4 HTML 表单

1. form表单 <form>标签: 用于创建一个表单, 通过表单, 用户可以向网站提交数据. 表单可以包含文本输入字段, 复选框, 单选按钮, 下拉列表, 提交按钮等等. 当用户提交表单时, 表单数据会发送到服务器进行处理.action属性: 应指向一个能够处理表单数据的服务器端脚本或UR…

Zabbix监控系统

目录 一、zabbix简介 1.1 zabbix 是什么&#xff1f; 1.2 zabbix 监控原理 二、安装zabbix 5.0 2.1 部署 zabbix 服务端 2.2 部署 zabbix 客户端 2.3 自定义监控内容 作为一个云计算行业从业人员&#xff0c;需要会使用监控系统查看服务器状态以及网站流量指标&#xff…

一篇文章看懂Apipost Mock功能怎么用

在接口开发过程中&#xff0c;Mock功能可以帮助开发者快速测试和验证接口的正确性和稳定性&#xff0c;以便快速迭代和修复问题。Apipost推出智能Mock功能&#xff0c;可以在智能期望中填写一些触发条件&#xff0c;开启后&#xff0c;Apipost会根据已设置的触发条件&#xff0…

口-肠-脑轴与精神健康的关系

谷禾健康 在个体中&#xff0c;每个微生物栖息地都表现出独特的微生物种群模式。迄今为止&#xff0c;关于微生物组相关疾病的研究主要集中在器官特异性微生物组上。然而&#xff0c;器官间的微生物网络正逐渐成为生理功能和病理过程中的重要调节因子和治疗机会。 在正常情况下…

mybatis打印sql语句出现多余的limit关键字

1、事情起因 在项目中使用了PageHelper分页插件&#xff0c;由于需求特殊&#xff0c;需要自定义分页&#xff0c;代码编写完成后&#xff0c;事故出现了。 前端传参: {pageNum: 1,pageSize: 10, }已知表中数据10条&#xff0c;但是每次分页查询只有10条数据&#xff0c;排查…