Spring注解补充(一)
注解补充
挑一些常用,但是深入不多的总结一下。
Bean的声明周期
在@Bean注解中,添加init属性
和destroy属性
@Bean(initMethod = "initMethod", destroyMethod = "destroyMethod")
public User user() {
return new User();
}
使用@PostContructor
和@PreDestory
官网描述:
官网说这 JSR-250
的@PostConstruct
和@PreDestroy
注解通常被认为是在现代Spring应用程序中接收生命周期回调的最佳实践。
代码实现:
自己定义初始化和结束的方法,分别加上两个注解。
@PostConstruct
public void initJSR250(){
System.out.println("User.initJSR250");
}
@PreDestroy
public void destroyJSR250(){
System.out.println("User.destroyJSR250");
}
实现InitializingBean
以及DisposableBean
接口
代码实现:
public class User implements InitializingBean, DisposableBean {
......
@Override
public void destroy() throws Exception {
System.out.println("User.destroy ---- bean");
}
@Override
public void afterPropertiesSet() throws Exception {
System.out.println("User.afterPropertiesSet ----- bean");
}
}
这些声明周期的注解、方法或者属性有着明显的先后顺序。
简单测试:
/*
* Copyright (c) 2006, 2023, wuyahan 编写
*
*/
package cn.lele.entity;
import lombok.AllArgsConstructor;
import lombok.Data;
import org.springframework.beans.factory.DisposableBean;
import org.springframework.beans.factory.InitializingBean;
import javax.annotation.PostConstruct;
import javax.annotation.PreDestroy;
/**
* <p>Project: ssm - User
* <p>Powered by wuyahan On 2023-02-24 09:36:11
*
* @author wuyahan [tianwenle2000@163.com]
* @version 1.0
* @since 17
*/
@Data
@AllArgsConstructor
public class User implements InitializingBean, DisposableBean {
private int id;
private String name;
private boolean sex;
public User(){
System.out.println("User类的构造方法执行");
}
public void save(){
System.out.println("User.save方法执行了");
}
@PostConstruct
public void initJSR250(){
System.out.println("User.initJSR250---------->@PostConstruct");
}
@PreDestroy
public void destroyJSR250(){
System.out.println("User.destroyJSR250----------->@PreDestroy");
}
public void initMethod(){
System.out.println("User.initMethod------->@Bean属性");
}
public void destroyMethod(){
System.out.println("User.destroyMethod-------->@Bean属性");
}
@Override
public void destroy() throws Exception {
System.out.println("User.destroy ----> 实现Spring提供的接口");
}
@Override
public void afterPropertiesSet() throws Exception {
System.out.println("User.afterPropertiesSet ----> 实现Spring提供的接口");
}
}
运行测试:
public class Main {
public static void main(String[] args) {
AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class);
User user = context.getBean(User.class);
user.save();
context.close();
}
}
运行结果:
可以发现执行顺序: JSR250的两个注解
> 实现Spring提供的接口
> 在@Bean中的属性
简单查看源码
自己构建Spring源码项目真的很好用,阅读起来也方便,还能自己加注释。
当前版本: Spring 5.3.x
将断点打在设置的Bean属性的初始化方法上,通过Debug的调用栈可以发现:
继续把Debug调试在1800行:
当beanName为user时,JSR250
已经执行。
执行下一步:
可以发现 实现Spring的初始化和关闭的接口
以及给Bean添加初始化方法和关闭的方法
是在一起执行的。
继续跟进1800行的代码,给这两行代码打上断点:
继续Debug:
执行下一步:
继续执行下一步:
可以发现Spring提供的两个接口执行顺序在自己给Bean添加属性的上面,所以先执行实现那两个接口的方法。
继续给User的JSR250
注解的初始化方法加上断点:
继续执行下一步:
可以发现在1796行的代码就是JSR250注解的方法的入口。
所以 对比三个初始化方法的源码位置,可以发现:JSR250
> 实现Spring的两个接口
> Bean属性添加方法
。
额外补充:在1796行代码继续跟进,发现里面是对BeanPostProcessor的处理。这个后面再继续补充。
@Import导入
基本导入
注入一个类:
public class User {
}
在配置类上导入:
@ComponentScan("cn.lele")
@Import(User.class)
public class AppConfig {
}
输出容器中Bean的名字:
public class Main {
public static void main(String[] args) {
AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class);
for (String name : context.getBeanDefinitionNames()) {
System.out.println("name = " + name);
}
}
}
实现ImportSelector接口
在配置类上导入:
@ComponentScan("cn.lele")
@Import({User.class, MySelector.class})
public class AppConfig {
}
查看导入Bean的名字:
实现MyImportBeanDefinitionRegistar接口
我们可以在ioc容器创建时,管理并且注册bean。通过实现MyImportBeanDefinitionRegistar
接口。
在配置类上导入:
@ComponentScan("cn.lele")
@Import({MyImportBeanDefinitionRegistrar.class, MySelector.class})
public class AppConfig {
}
查看容器中所有Bean的名字:
@CompnentScan包扫描
这个注解还是有一些小坑的......
包含include小坑
CompnentScan注解
默认开启自动检测带有@Component
、@Repository
、@Service
、@Controller
。所以就会有一个坑,当我们默认去include的时候,如果不关闭默认过滤,添加将不会发挥作用。
当我们只想加载Service注解和Controller注解的时候,如果不取消默认的扫描,包含将会失效:
取消默认扫描:
排除不受影响
自定义扫描CUSTOM
自定义扫描需要实现TypeFilter
接口
public class MyTypeFilter implements TypeFilter {
@Override
public boolean match(MetadataReader metadataReader, MetadataReaderFactory metadataReaderFactory) throws IOException {
// 获取Bean的名字
String className = metadataReader.getClassMetadata().getClassName();
// 包含Dao的允许加载
if (className.toUpperCase().contains("DAO")){
return true; //允许加载
}
return false; // 不允许加载
}
}
type选择为FilterType.CUSTOM
@Configuration
@ComponentScan(
value = "cn.lele" ,
includeFilters = @ComponentScan.Filter(type = FilterType.CUSTOM,classes = {MyTypeFilter.class}),
useDefaultFilters = false // 包含都需要关闭这个默认过滤
)
public class AppConfig {
}
运行结果:
Spring注解补充(一)的更多相关文章
- spring注解式参数校验
很痛苦遇到大量的参数进行校验,在业务中还要抛出异常或者返回异常时的校验信息,在代码中相当冗长,今天我们就来学习spring注解式参数校验. 其实就是:hibernate的validator. 开始啦. ...
- spring注解-@Autowired。@Resource。@Service
Spring的@Autowired注解.@Resource注解和@Service注解 什么是注解 传统的Spring做法是使用.xml文件来对bean进行注入或者是配置aop.事物,这么做有两个缺点: ...
- 0、Spring 注解驱动开发
0.Spring注解驱动开发 0.1 简介 <Spring注解驱动开发>是一套帮助我们深入了解Spring原理机制的教程: 现今SpringBoot.SpringCloud技术非常火热,作 ...
- Spring 注解无效 空指针异常
对于Java开发,现在Spring已成为一种生态,使用Spring成为一种享受,Spring的使用让开发变得更加便捷. 而Spring好用归好用,若不清楚里面的工作原理,使用的时候难免会踩到一些坑. ...
- 浅尝Spring注解开发_AOP原理及完整过程分析(源码)
浅尝Spring注解开发_AOP原理及完整过程分析(源码) 浅尝Spring注解开发,基于Spring 4.3.12 分析AOP执行过程及源码,包含AOP注解使用.AOP原理.分析Annotation ...
- Spring注解开发_Spring容器创建概述
浅尝Spring注解开发_Spring容器创建概述 浅尝Spring注解开发,基于Spring 4.3.12 概述Spring容器创建的过程,包括12个方法的执行 浅尝Spring注解开发_自定义注册 ...
- spring注解源码分析--how does autowired works?
1. 背景 注解可以减少代码的开发量,spring提供了丰富的注解功能.我们可能会被问到,spring的注解到底是什么触发的呢?今天以spring最常使用的一个注解autowired来跟踪代码,进行d ...
- Spring注解
AccountController .java Java代码 1. /** 2. * 2010-1-23 3. */ 4. packag ...
- spring 注解的优点缺点
注解与XML配置的区别 注解:是一种分散式的元数据,与源代码耦合. xml :是一种集中式的元数据,与源代码解耦. 因此注解和XML的选择上可以从两个角度来看:分散还是集中,源代码耦合/解耦. 注解的 ...
- spring注解说明之Spring2.5 注解介绍(3.0通用)
spring注解说明之Spring2.5 注解介绍(3.0通用) 注册注解处理器 方式一:bean <bean class="org.springframework.beans.fac ...
随机推荐
- day27 CSS浮动、溢出 & js基本语法 & DOM文档流操作
接day26CSS=>CSS定位 overflow属性 值 描述 示例 visible 默认值,内容不会被修剪,会呈现在元素框之外 hidden 内容会被修剪,并且其余内容是不可见的 overf ...
- Qwt开发笔记(一):Qwt简介、下载以及基础demo工程模板
前言 QWT开发笔记系列整理集合,这是目前使用最为广泛的Qt图表类(Qt的QWidget代码方向只有QtCharts,Qwt,QCustomPlot),使用多年,系统性的整理,本系列旨在系统解说并 ...
- 【软考-中级-数据库相关知识】03、SQL语言
SQL语言 查询 允许出现聚集函数的是:select子句和having子句 权限管理 语法:GRANT XXX ON TABLE TO USER WITH GRANT OPTION 存储过程和函数 函 ...
- 【Java SE】课程目录
〇.课程简介 一.前言入门 二.数据类型.运算符.方法 三.流程控制 四.Idea.方法 五.数组 六.类.对象.封装.构造 七.Scanner类.Random类.ArrayList类 八.Strin ...
- 【消息队列面试】6-10:Rebalance机制、副本同步机制、架构设计、zk的作用、kafka的高性能
六.简述kafka的Rebalance[偏向实战,有难度] 1.背景 kafka日志:在消息量大.高并发时,经常会出现rebalance中 rebalance会影响kafka性能,会阻塞partiti ...
- C++日期和时间编程总结
一,概述 二,C-style 日期和时间库 2.1,数据类型 2.2,函数 2.3,数据类型与函数关系梳理 2.4,时间类型 2.4.1,UTC 时间 2.4.2,本地时间 2.4.3,纪元时间 2. ...
- 填坑日志-云网络智慧课堂双网卡Mac地址读取错误的问题及解决
云网络智慧课堂的双网卡问题记录及解决方案 教师端 其实这里双网卡的问题一直没有解决,分为了两部分,一部分是教师端,一部分是学生端.症状类似,问题也类似,都是在设计之初因为硬件限制可能没有考虑到双网卡的 ...
- PhaApi NOTORM 实现分表分库
通过自增ID取模要分表的数量,便可得到表名.例如log表分成100张表:log_1,log2...,log100. 每次数据库CURD都先通过获取ID分配到相对应的表,例如:id=66,取模后的结果是 ...
- Error: Could not get apiVersions from Kubernetes
问题 部署pod时遇到问题 # helm install chart.tgz Error: Could not get apiVersions from Kubernetes: unable to r ...
- Spark详解(01) - Scala编程语言
Spark详解(01) - Scala编程语言概述 Scala官网:https://www.scala-lang.org/ 什么是Scala 从英文的角度来讲,Scala并不是一个单词,而是Scala ...