Spring注解开发之组件注册(二)
创始人
2024-05-31 04:06:54
0

Spring注解开发之组件注册(一)

5.@Import 给容器导入一个组件

给容器中注册组件

一、包扫描 + 组件标注注解(@Controller/@Service/@Repository/@Component) [自己写的类]

二、@Bean [导入的第三包里面的组件]

三、@Import [快速给容器中导入组件] (@Import{Color.class,Red.class})

    1.@Import(要导入到容器中的组件),容器中就会自动注册这个组件,id默认是组件全名

在这里插入图片描述

​ 2. @ImportSelector: 返回需要导入组建的全类名数组

// 自定义逻辑,返回需要导入的组件
public class MyImportSelector implements ImportSelector {// 返回值就是要导入到容器中的组件的全类名// AnnotationMetadata:当前标注@Import注解类的所有信息@Overridepublic String[] selectImports(AnnotationMetadata importingClassMetadata) {return new String[]{"com.lv.pojo.Blue", "com.lv.pojo.Yellow"};// 返回的类注入容器}
}

​ 3.使用ImportBeanDefinitionRegister 手动注册bean到容器中

public class MyImportBeanDefinitionRegistrar implements ImportBeanDefinitionRegistrar {/*** AnnotationMetadata:当前类的注解信息* BeanDefinitionRegistry:注册类* 把所有需要添加到容器中的bean,调用BeanDefinitionRegistry.registerBeanDefinition手动注册*/@Overridepublic void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {// 查询容器中是否存在red组件boolean red = registry.containsBeanDefinition("com.xjhqre.pojo.Red");if (red) {// 指定bean的定义信息(bean的类型,bean的scope)BeanDefinition beanDefinition = new RootBeanDefinition(Purple.class);// 第一个参数指定bean的idregistry.registerBeanDefinition("purple", beanDefinition);}}
}

四、使用Spring提供的FactoryBean (工厂Bean)

​ 1.默认获取到的是工厂bean调用getObject创建的对象

​ 2.要获取工厂Bean本身,我们需要给id前面加一个& 即 &colorFactoryBean

// 创建一个Spring定义的FactoryBean
public class ColorFactoryBean implements FactoryBean {// 返回一个Color对象,这个对象会添加到容器中@Overridepublic Color getObject() throws Exception {return new Color();}@Overridepublic Class getObjectType() {return Color.class;}// 返回是否为单例,// true 单实例  在容器中保存一份// false,则每次创建时调用getObject()方法@Overridepublic boolean isSingleton() {return true;}
}

在配置类中导入这个组件!!!

@Test
public void test5() {AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(MainConfig4.class);String[] beanDefinitionNames = applicationContext.getBeanDefinitionNames();// 工厂bean获取的是调用getObject创建的对象Object colorFactoryBean = applicationContext.getBean("colorFactoryBean");System.out.println(colorFactoryBean.getClass());// 如果想要拿到colorFactoryBean对象,则需要在传入的id前加一个&标识Object colorFactoryBean2 = applicationContext.getBean("&colorFactoryBean");System.out.println(colorFactoryBean2.getClass());
}

6.Bean生命周期

即bean从创建 ----> 初始化 -----> 销毁的过程

容器管理bean的生命周期;

我们可以自定义初始化和销毁方法,容器在bean进行进行到当前生命周期可以自定义初始化和销毁方法

构造(对象创建)

  • 单实例:在容器启动的时候创建对象
  • 多实例:在每次获取的时候创建对象

初始化

  • 对象完成创建,并赋值好后,调用初始化方法

销毁

  • 单实例:在容器关闭的时候销毁

  • 多实例:容器不会管理这个bean,所以不会调用销毁方法(初始化会在第一次使用这个对象时才会创建)

在这里插入图片描述

初始化和销毁方法:

  1. 通过@Bean注解指定init-method和destroy-method

在这里插入图片描述
在这里插入图片描述

  1. 实现InitializingBean和DisposableBean接口,重写里面的destroy和afterPropertiesSet方法

    1. 通过让bean实现InitializingBean接口来定义初始化逻辑
    2. 通过让bean实现DisposableBean接口来定义销毁逻辑
    3. InitializingBean中有一个方法afterPropertiesSet,该方法在bean创建并赋值后调用
  2. 使用@PostConstruct和@PreDestroy注解

    1. @PostConstruct:在bean创建完成并且属性赋值完成后进行初始化
    2. @PreDestroy:在容器销毁bean之前执行
  3. BeanPostProcessor:接口,bean后置处理器,在bean初始化前后进行一些处理

    1. postProcessBeforeInitialization:在初始化之前执行
    2. postProcessAfterInitialization:在初始化之后执行

BeanPostProcessor执行原理

1.执行populateBean(beanName, mbd, instanceWrapper);给bean进行属性赋值

2.开始initializeBean初始化bean

​ 1.先执行applyBeanPostProcessorsBeforeInitialization(wrappedBean, beanName);,遍历得到容器中所有的BeanPostProcessor;挨个执行beforeInitialization,一但返回null,跳出for循环,不会执行后面的BeanPostProcessor.postProcessorsBeforeInitialization
​ 2.然后执行invokeInitMethods(beanName, wrappedBean, mbd);执行自定义初始化
​ 3.最后执行applyBeanPostProcessorsAfterInitialization(wrappedBean, beanName);

7.@Value 属性赋值

在这里插入图片描述

在这里插入图片描述

8.自动装配

Spring利用依赖注入(DI)完成对IOC容器中各个组件的依赖关系赋值

  1. @Autowired:自动注入
    (1)默认优先按照类型去容器中去找对应的组件: applicationContext.getBean(BookDao.class);如果找到了则进行赋值;
    (2)如果找到了多个相同类型的组件,再将属性的名称作为组件的id去容器中查找applicationContext.getBean(“bookDao”)
    (3)@Qualifier(“bookDao”):使用 @Qualifier 指定需要装配的组件的id,而不是使用属性名
    (4)自动装配默认一定要将属性赋值好,没有就会报错,可以使用@Autowired(required = false);来设置为非必须的
    (5)可以利用@Primary:让Spring在进行自动装配的时候,默认使用首选的bean,也可以继续使用@Qualifier(“bookDao”)来明确指定需要装配的bean的名字

  2. Spring还支持使用**@Resource(JSR250)**和@Inject(JSR330) [Java规范注解]

    1. @Resource 可以和@Autowired一样自动装配功能,默认是按照组件名称进行装配的;没有能支持@primary功能没有支持@Autowired(required = false)

    2. @Inject:需要导入java.inject的包,和@Autowired的功能一样

在这里插入图片描述

  1. @Autowired:Spring定义的 @Resource和@Inject都是java规范的AutowiredAnnotationBeanPostProcessor:解析完成自动装配 以上注解都装配进去了

  2. 方法、构造器位置的自动装配

    1. 放在方法上,@Bean+方法参数:参数从容器中获取 ,默认不写@Autowired在这里插入图片描述

    2. 标在构造器上只有一个有参构造器,这个有参构造器的@Autowired可以省略,其参数位置的组件还是可以自动从容器中获取 默认加在ioc容器中的组件,容器会调用无参构造器创建对象,再进行初始化赋值等操作 在这里插入图片描述

  3. Aware注入Spring底层组件 原理

    1. Aware 接口,提供了类似回调函数的功能
    2. 自定义组件想要使用Spring 容器底层的一些组件(ApplicationContext,BeanFactory);自定义组件需要实现xxxAware接口;在创建对象的时候,会调用接口规定的方法注入相关组件
    3. 原理:通过对应的Processor进行处理的 ApplicationContextAware => ApplicationContextAwareProcessor 后置处理器,在创建完bean后,看见bean实现了相关Aware接口,将组件传进来

9.@Profile环境搭建

Spring我们提供的可以根据当前环境,动态的激活和切换一系列组建的功能

​ 开发、生产、测试环境 数据源的不同

@Profile 指定组件在那个环境的情况下才能被注册到容器中,不指定,任何环境下都能注册这组件
在这里插入图片描述

我们需要在项目的src/main/resources目录下新建一个配置文件,例如dbconfig.properties,在其中写上数据库连接的相关信息,如下所示。

db.user=root
db.password=123456
db.driverClass=com.mysql.jdbc.Driver
import javax.sql.DataSource;import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.EmbeddedValueResolverAware;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.PropertySource;
import org.springframework.util.StringValueResolver;import com.mchange.v2.c3p0.ComboPooledDataSource;/**** @author liayun**/
@PropertySource("classpath:/dbconfig.properties") // 加载外部的配置文件
@Configuration
public class MainConfigOfProfile implements EmbeddedValueResolverAware {@Value("${db.user}")private String user;private StringValueResolver valueResolver;private String dirverClass;@Bean("testDataSource")public DataSource dataSourceTest(@Value("${db.password}") String pwd) throws Exception {ComboPooledDataSource dataSource = new ComboPooledDataSource();dataSource.setUser(user);dataSource.setPassword(pwd);dataSource.setJdbcUrl("jdbc:mysql://localhost:3306/test");dataSource.setDriverClass(dirverClass);return dataSource;}@Bean("devDataSource")public DataSource dataSourceDev(@Value("${db.password}") String pwd) throws Exception {ComboPooledDataSource dataSource = new ComboPooledDataSource();dataSource.setUser(user);dataSource.setPassword(pwd);dataSource.setJdbcUrl("jdbc:mysql://localhost:3306/ssm_crud");dataSource.setDriverClass(dirverClass);return dataSource;}@Bean("prodDataSource")public DataSource dataSourceProd(@Value("${db.password}") String pwd) throws Exception {ComboPooledDataSource dataSource = new ComboPooledDataSource();dataSource.setUser(user);dataSource.setPassword(pwd);dataSource.setJdbcUrl("jdbc:mysql://localhost:3306/scw_0515");dataSource.setDriverClass(dirverClass);return dataSource;}// 继承EmbeddedValueResolverAware接口// 他的实现就是直接用${db.driverClass}获取配置文件的值@Overridepublic void setEmbeddedValueResolver(StringValueResolver resolver) {this.valueResolver = resolver;dirverClass = valueResolver.resolveStringValue("${db.driverClass}");}}

相关内容

热门资讯

122.(leaflet篇)l... 听老人家说:多看美女会长寿 地图之家总目录(订阅之前建议先查看该博客) 文章末尾处提供保证可运行...
育碧GDC2018程序化大世界... 1.传统手动绘制森林的问题 采用手动绘制的方法的话,每次迭代地形都要手动再绘制森林。这...
育碧GDC2018程序化大世界... 1.传统手动绘制森林的问题 采用手动绘制的方法的话,每次迭代地形都要手动再绘制森林。这...
Vue使用pdf-lib为文件... 之前也写过两篇预览pdf的,但是没有加水印,这是链接:Vu...
PyQt5数据库开发1 4.1... 文章目录 前言 步骤/方法 1 使用windows身份登录 2 启用混合登录模式 3 允许远程连接服...
Android studio ... 解决 Android studio 出现“The emulator process for AVD ...
Linux基础命令大全(上) ♥️作者:小刘在C站 ♥️个人主页:小刘主页 ♥️每天分享云计算网络运维...
再谈解决“因为文件包含病毒或潜... 前面出了一篇博文专门来解决“因为文件包含病毒或潜在的垃圾软件”的问题,其中第二种方法有...
南京邮电大学通达学院2023c... 题目展示 一.问题描述 实验题目1 定义一个学生类,其中包括如下内容: (1)私有数据成员 ①年龄 ...
PageObject 六大原则 PageObject六大原则: 1.封装服务的方法 2.不要暴露页面的细节 3.通过r...
【Linux网络编程】01:S... Socket多进程 OVERVIEWSocket多进程1.Server2.Client3.bug&...
数据结构刷题(二十五):122... 1.122. 买卖股票的最佳时机 II思路:贪心。把利润分解为每天为单位的维度,然后收...
浏览器事件循环 事件循环 浏览器的进程模型 何为进程? 程序运行需要有它自己专属的内存空间࿰...
8个免费图片/照片压缩工具帮您... 继续查看一些最好的图像压缩工具,以提升用户体验和存储空间以及网站使用支持。 无数图像压...
计算机二级Python备考(2... 目录  一、选择题 1.在Python语言中: 2.知识点 二、基本操作题 1. j...
端电压 相电压 线电压 记得刚接触矢量控制的时候,拿到板子,就赶紧去测各种波形,结...
如何使用Python检测和识别... 车牌检测与识别技术用途广泛,可以用于道路系统、无票停车场、车辆门禁等。这项技术结合了计...
带环链表详解 目录 一、什么是环形链表 二、判断是否为环形链表 2.1 具体题目 2.2 具体思路 2.3 思路的...
【C语言进阶:刨根究底字符串函... 本节重点内容: 深入理解strcpy函数的使用学会strcpy函数的模拟实现⚡strc...
Django web开发(一)... 文章目录前端开发1.快速开发网站2.标签2.1 编码2.2 title2.3 标题2.4 div和s...