【SpringⅢ】Spring 的生命周期

目录

🥪1 Bean 的作用域

🥩1.1 singleton:单例模式

🍙1.2 prototype:原型模式

🍱1.3 Bean 的其他作用域

🍜2 Spring 生命周期(执行流程)

🥘2.1 启动容器

🍲 2.2 读取配置文件完成 Bean 初始化(实例化)

🎂2.3 将 Bean 对象放到容器中

🫕2.4 注入 Bean 属性(给当前类的属性DI,进行赋值)

🧁3 Bean 生命周期


1 Bean 的作用域

指定程序中某个变量的使用范围就叫做作用域。而 Bean 的作用域指的是 Bean 在 Spring 整个框架的行为模式。比如单例模式,就意味着 Bean 在整个 Spring 中只有一份,为全局共享的,如果有人修改这个值,那么另一个人拿到的,就是被修改过后的值。

1.1 singleton:单例模式

该作用域下的 Bean 在 IoC 容器中只存在一个实例,所以获取 Bean 或者注入 Bean(通过 @Autowired 注入)都是同一个对象

package com.java.demo.Model;public class User {private String name;private int Id;@Overridepublic String toString() {return "User{" +"name='" + name + '\'' +", Id=" + Id +'}';}public String getName() {return name;}public void setName(String name) {this.name = name;}public int getId() {return Id;}public void setId(int id) {Id = id;}
}

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.setName("唐三藏");user.setId(11);return user;}
}

package com.java.demo.Controller;import com.java.demo.Model.User;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;@Controller
public class UserController {@Autowiredprivate User user;public void doMethod(){User user1 = user;System.out.println("UserController 修改之前 :User -> " + user);user1.setName("如来佛祖");System.out.println("UserController 修改之后 :User -> " + user);}
}

package com.java.demo.Controller;import com.java.demo.Model.User;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;@Controller
public class UserController2 {@Autowiredprivate User user;public void doMethod(){System.out.println("UserController2 :User -> " + user);}
}

package com.java.demo;import com.java.demo.Controller.UserController;
import com.java.demo.Controller.UserController2;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;/**
* 启动类*/
public class App {public static void main(String[] args) {ApplicationContext context =new ClassPathXmlApplicationContext("spring-config.xml");UserController userController = context.getBean("userController", UserController.class);userController.doMethod();UserController2 userController2 = context.getBean("userController2", UserController2.class);userController2.doMethod();}
}

输出:

UserController 修改之前 :User -> User{name='唐三藏', Id=11}
UserController 修改之后 :User -> User{name='如来佛祖', Id=11}
UserController2 :User -> User{name='如来佛祖', Id=11}

可以发现,单例模式中,user的姓名被修改了。

1.2 prototype:原型模式

在该作用域下,每次 Bean 请求都会创建新的实例,因此获取 Bean 以及注入 Bean 都是新的对象实例。

可以使用 @Scope 标签来设置作用域。@Scope 既可以修饰方法也可以修饰类,有以下两种写法:

1. 直接设置:@Scope("prototype")

2. 枚举设置:@Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)

package com.java.demo.Model;import org.springframework.beans.factory.config.ConfigurableBeanFactory;
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") // 原型模式/多例模式// 亦可写成:@Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)public User getUser(){User user = new User();user.setName("唐三藏");user.setId(11);return user;}
}

输出:

UserController 修改之前 :User -> User{name='唐三藏', Id=11}
UserController 修改之后 :User -> User{name='如来佛祖', Id=11}
UserController2 :User -> User{name='唐三藏', Id=11}

1.3 Bean 的其他作用域

只适用于 Spring MVC 项目(Spring Web):

request:请求作用域。每次 http 请求都会创建新的 Bean 实例。一次 http 请求和响应共享                    Bean

session:会话作用域。一个 Http 会话共享一个 Bean。

application:应用作用域。表示的是一个 Context 容器共享一个作用域。

只适用于 websocket :

websocket 作用域

2 Spring 生命周期(执行流程)

2.1 启动容器

 2.2 读取配置文件完成 Bean 初始化(实例化)

2.3 将 Bean 对象放到容器中

这里只会扫描包路径上的类并且使用 Spring 的注解才可以被放到容器中。

扫描 com.java.demo 包下的 Spring 注解,像 @Controller、@Service、@Component、@Repository 

2.4 注入 Bean 属性(给当前类的属性DI,进行赋值)

如果 Bean 对象需要使用其他 Bean 对象作为属性,可以使用注解:@Autowired、@Resource

3 Bean 生命周期

Bean 的生命周期指的是 Bean 从诞生到销毁的整个生命过程,这样的过程就叫做生命周期,可以大致分为以下 5 个部分:

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

2. 设置 Bean 属性(进行依赖注入,将依赖的 Bean 赋值到当前类的属性上)

3. Bean 初始化

           实现了各种 Aware 通知的⽅法,如 BeanNameAware、BeanFactoryAware、                         ApplicationContextAware 的接⼝⽅法;

           执⾏ BeanPostProcessor 初始化前置⽅法

           执⾏ @PostConstruct 初始化⽅法,依赖注⼊操作之后被执⾏;

           执⾏⾃⼰指定的 init-method ⽅法(如果有指定的话);

           执⾏ BeanPostProcessor 初始化后置⽅法

4. 使用 Bean

5. 销毁 Bean

           销毁容器的各种⽅法,如 @PreDestroy、DisposableBean 接⼝⽅法、destroy-                       method。

 

package com.java.demo;import com.java.demo.Model.User;
import org.springframework.beans.factory.BeanNameAware;
import org.springframework.beans.factory.annotation.Autowired;import javax.annotation.PostConstruct;
import javax.annotation.PreDestroy;public class BeanLifeComponent implements BeanNameAware {@Autowiredprivate User user;@Overridepublic void setBeanName(String s) {System.out.println("执行了 BeanNameAware -> " + s);}@PostConstructpublic void doPostConstruct(){System.out.println("执行了 @PostConstruct");System.out.println(user.toString());}public void myInit(){System.out.println("执行了 myInit");}@PreDestroypublic void doPreDestroy(){System.out.println("执行了 @PreDestroy");}public void sayHi(){System.out.println("使用了 Bean");}
}
package com.java.demo;import org.springframework.context.ApplicationContext;
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();}
}

 

输出:

执行了 BeanNameAware -> myBean
执行了 @PostConstruct
User{name='唐三藏', Id=11}
执行了 myInit
使用了 Bean
执行了 @PreDestroy


 

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

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

相关文章

Logback日志输入到xxl-job日志中

前言 ​ 在xxl-job的任务中,如果需要把日志输入到xxl-job的日志文件里,需要使用XxlJobHelper.log来记录日志,这种方式才能在任务执行详情里面看到对应的日志。而有时候 习惯用Slf4j来记录日志,而通过slf4j打印的日志没办法在xxl-…

[小尘送书-第二期]《从零开始读懂量子力学》由浅入深,解释科学原理;从手机到超导,量子无处不在;从微观到宏观,遐想人生的意义!

大家好,我是小尘,欢迎关注,一起交流学习!欢迎大家在CSDN后台私信我!一起讨论学习,讨论如何找到满意的工作! 本文目录 一、前言二、作者简介三、内容简介四、抽奖方式五、名家推介写在最后 一、前…

小程序 表单验证

使用 WxValidate.js 插件来校验表单数据 常用实例方法 名称返回类型描述checkForm(e)boolean验证所有字段的规则,返回验证是否通过。valid()boolean返回验证是否通过。size()number返回错误信息的个数。validationErrors()array返回所有错误信息。addMethod(name…

[containerd] 在Windows上使用IDEA远程调试containerd, ctr, containerd-shim

文章目录 1. containerd安装2. 源码编译3. 验证编译的二进制文件是否含有调试需要的信息3.1. objdump工具验证3.2. file工具验证3.3. dlv工具验证 4. debug 1. containerd安装 [Ubuntu 22.04] 安装containerd 2. 源码编译 主要步骤如下: 1、从github下载containe…

GO学习之 通道(Channel)

GO系列 1、GO学习之Hello World 2、GO学习之入门语法 3、GO学习之切片操作 4、GO学习之 Map 操作 5、GO学习之 结构体 操作 6、GO学习之 通道(Channel) 文章目录 GO系列前言一、Channel 简介二、初始化通道2.1 无缓冲通道2.2 有缓冲通道2.3 单向通道 三、通道操作3.1 关闭通道…

[Java] 单例设计模式详解

模式定义:保证一个类只有一个实例,并且提供一个全局访问点,时一种创建型模式 使用场景:重量级的对象,不需要多个实例,如线程池,数据库连接池 单例设计模式的实现 1.懒汉模式:延迟…

第一次后端复习整理(JVM、Redis、反射)

1. JVM 文章仅为自身笔记 详情查看一篇文章掌握整个JVM,JVM超详细解析!!! 1.1 什么是JVM jvm是Java虚拟机 1.2 Java文件的编译过程 程序员编写代码形成.java文件经过javac编译成.class文件再通过JVM的类加载器进入运行时数据…

【141. 环形链表】

来源:力扣(LeetCode) 描述: 给你一个链表的头节点 head ,判断链表中是否有环。 如果链表中有某个节点,可以通过连续跟踪 next 指针再次到达,则链表中存在环。 为了表示给定链表中的环&#x…

JVM | 基于类加载的一次完全实践

引言 我在上篇文章:JVM | 类加载是怎么工作的 中为你介绍了Java的类加载器及其工作原理。我们简单回顾下:我用一个易于理解的类比带你逐步理解了类加载的流程和主要角色:引导类加载器,扩展类加载器和应用类加载器。并带你深入了解…

剑指offer12 矩阵中的路径 13 机器人的运动范围 34.二叉树中和为某一值得路径

class Solution { public:bool exist(vector<vector<char>>& board, string word) {int rowboard.size(),colboard[0].size();int index0,i0,j0;if(word.size()>row*col) return 0;//vector<vector<int>> visit[row][col];//标记当前位置有没有…

算法之归并排序算法

归并&#xff08;Merge&#xff09;排序法是将两个&#xff08;或两个以上&#xff09;有序表合并成一个新的有序表&#xff0c;即把待排序序列 分为若干个子序列&#xff0c;每个子序列是有序的。然后再把有序子序列合并为整体有序序列。 public class MergeSortTest {public …

到底什么是前后端分离

目录 Web 应用的开发主要有两种模式&#xff1a; 前后端不分离 前后端分离 总结 Web 应用的开发主要有两种模式&#xff1a; 前后端不分离 前后端分离 理解它们的区别有助于我们进行对应产品的测试工作。 前后端不分离 在早期&#xff0c;Web 应用开发主要采用前后端不…

linux系统下(centos7.9)安装Jenkins全流程

一、卸载历史版本 # rpm卸载 rpm -e jenkins# 检查是否卸载成功 rpm -ql jenkins# 彻底删除残留文件 find / -iname jenkins | xargs -n 1000 rm -rf二、环境依赖安装 yum -y install epel-releaseyum -y install daemonize三、安装Jenkins Jenkins官网传送带&#xff1a; …

《零基础入门学习Python》第071讲:GUI的终极选择:Tkinter8

虽然我们能用 tkinter 设计不少东西了&#xff0c;但是不少同学还是感觉对这个界面编程掌控得还不够多&#xff0c;说白了&#xff0c;就是我们现在还没办法随心所欲的去绘制我们想要的界面&#xff0c;但是不瞒你说&#xff0c;今天的这一节课将会给你的人生乃至人生观带来翻天…

VB+access文档管理系统设计与实现

内容摘要 《文档管理系统》是采用VISAUL BASIC6.0开发的一个数据库管理系统。本设计说明书主要讲述了VISAUL BASIC6.0的基本功能及设计方法。紧接着以本系统为例,逐一介绍开发本系统系统的步骤:系统分析、系统设计、系统实现、系统维护。在系统分析中先后用数据流图、数据字…

Android程序CPU使用大的异常分析

程序出现CPU使用过高的问题&#xff0c;如果能够重现&#xff0c;就比较好办了&#xff0c;可以top命令查看各线程的cpu使用&#xff0c;定位到线程。 以下是问国内某AI的答案 在Android应用中&#xff0c;如果某个应用消耗了大量的CPU资源&#xff0c;可以采取以下方法进行分…

分布式id、系统id、业务id以及主键之间的关系

推荐 连分布式ID都理解不了&#xff0c;你是刚培训出来冒充面试官的吧 1 分布式id、系统id、业务id以及主键之间的关系 分布式ID、系统ID、业务ID和主键的关系&#xff1a; 分布式ID&#xff1a;在分布式系统中&#xff0c;由于存在多个独立的节点&#xff0c;为了保证每个节…

苍穹外卖-day07

苍穹外卖-day07 本项目学自黑马程序员的《苍穹外卖》项目&#xff0c;是瑞吉外卖的Plus版本 功能更多&#xff0c;更加丰富。 结合资料&#xff0c;和自己对学习过程中的一些看法和问题解决情况上传课件笔记 视频&#xff1a;https://www.bilibili.com/video/BV1TP411v7v6/?sp…

CSS盒子模型(HTML元素布局)

CSS盒子模型是一种用于描述HTML元素布局的模型&#xff0c;它将每个元素看作是一个矩形的盒子&#xff0c;每个盒子由内容、内边距、边框和外边距组成。 盒子模型包括以下几个部分&#xff1a; 内容区域&#xff08;Content&#xff09; 内容区域是盒子中实际显示内容的部分&am…

Rust vs Go:常用语法对比(五)

题图来自 Rust vs Go 2023[1] 81. Round floating point number to integer Declare integer y and initialize it with the rounded value of floating point number x . Ties (when the fractional part of x is exactly .5) must be rounded up (to positive infinity). 按规…