• Spring注解



    在这里插入图片描述
    图片来源: https://zhuanlan.zhihu.com/p/38208324

    尚硅谷视频地址:https://www.bilibili.com/video/BV1gW411W7wy

    核心思想:
    1、Spring容器在启动的时候,先会保存所有注册进来的Bean的定义信息;

    • xml注册bean
    • 注解注册Bean;@Service、@Component、@Bean、xxx

    2、Spring容器会合适的时机创建这些Bean

    • 用到这个bean的时候;利用getBean创建bean;创建好以后保存在容器中;
    • 统一创建剩下所有的bean的时候;finishBeanFactoryInitialization();

    3、后置处理器;BeanPostProcessor

    • 每一个bean创建完成,都会使用各种后置处理器进行处理;来增强bean的功能;
      • AutowiredAnnotationBeanPostProcessor:处理自动注入
      • AnnotationAwareAspectJAutoProxyCreator:来做AOP功能;

    4、事件驱动模型

    • ApplicationListener:事件监听;
    • ApplicationEventMulticaster:事件派发:

    Spring IOC

    @Configuration

    指定是一个配置类,自身也会被加入容器中

    @Bean

    添加组件

    @ComponentScan

    指定要扫描得包,还可以指定包含哪些组件、不包含哪些组件
    可通过实现TypeFilter来自定义过滤器,:@ComponentScan.Filter(type = FilterType.CUSTOM, classes = {MyTypeFilter.class}),其中MyTypeFilter是TypeFilter实现类

    @Scope

    设置单实例(默认)、多实例

    • ConfigurableBeanFactory.SCOPE_PROTOTYPE,多例
    • ConfigurableBeanFactory.SCOPE_SINGLETON,单例
    • org.springframework.web.context.WebApplicationContext.SCOPE_REQUEST,同一请求创建一个实例
    • org.springframework.web.context.WebApplicationContext.SCOPE_SESSION, 同一session创建一个实例

    @Lazy

    在容器启动时不创建,在第一次获取Bean时创建对象并初始化。

    @Conditional

    传入一个Condition数组,Condition接口:

    @FunctionalInterface
    public interface Condition {
    
        /**
         * ConditionContext:判断条件能使用的上下文环境
         * AnnotatedTypeMetadata:注释信息
         */
        boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata);
    
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10

    组件注册方法

    @Controller、@Service、@Repository、@Component

    包扫描 + 组件标注注解(@Controller、@Service、@Repository、@Component),这种方式只能注册自己写的类

    @Bean

    @Bean,可以注册第三方包的类

    @Import

    @Import:快速给容器中导入一个组件,id默认是全类名,除了可以导入一般的类,还可以导入以下内容用于注册组件:

    • ImportSelector:一个接口,返回需要导入的组件的全类名数组

      • 实现selectImports(AnnotationMetadata importingClassMetadata)方法
        • 参数importingClassMetadata:当前标注@Import注解类的所有信息
    • ImportBeanDefinitionRegistrar:一个接口,手动注册bean到容器中,可以自定义组件名称

      • 实现registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry)方法
        • 参数importingClassMetadata:当前标注@Import注解类的所有信息
        • 参数registry注册组件的类,可用于手动注册组件
      • 通过registry.registerBeanDefinition方法注册组件

    FactoryBean

    `使用Spring提供的FactoryBean(工厂Bean),getObject()添加组件,还可以定义组件类型、是否单例

    Bean生命周期

    描述

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

    容器管理bean的生命周期:
    1、对象创建

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

    2、初始化

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

    3、销毁

    • 单实例:在容器关闭的时候销毁
    • 多实例:容器不会管理这个bean,所以不会调用销毁方法

    自定义组件初始化和销毁方法

    我们可以自定义初始化和销毁方法

    初始化和销毁方法:

    • 通过@Bean注解指定init-method和destroy-method
    • 实现InitializingBeanDisposableBean接口,重写里面的destroy和afterPropertiesSet方法
    • 使用@PostConstruct@PreDestroy注解
      • PostConstruct:在bean创建完成并且属性赋值完成后进行初始化
      • PreDestroy:在容器销毁bean之前执行
    • BeanPostProcessor:接口,bean后置处理器,在bean初始化前后进行一些处理
      • postProcessBeforeInitialization:在初始化之前执行
      • postProcessAfterInitialization:在初始化之后执行

    BeanPostProcessor执行过程

    • 执行populateBean(beanName, mbd, instanceWrapper);给bean进行属性赋值
    • 开始initializeBean初始化bean
      • 先执行applyBeanPostProcessorsBeforeInitialization(wrappedBean, beanName);,遍历得到容器中所有的BeanPostProcessor;挨个执行beforeInitialization,一但返回null,跳出for循环,不会执行后面的BeanPostProcessor.postProcessorsBeforeInitialization
      • 然后执行invokeInitMethods(beanName, wrappedBean, mbd);执行自定义初始化
      • 最后执行applyBeanPostProcessorsAfterInitialization(wrappedBean, beanName);

    @Value

    使用@Value赋值,赋值方法:

    • 直接在@Value中写基本数值
    • 使用SpEL表达式:#{},如#{6+6}表示12
    • 使用${}:取出配置文件中的值(在运行环境变量里的值)

    @PropertySource

    例如:@PropertySource(“classpath:a.properties”),可以读取a.properties中的k-v值,之后使用${...}可以取出值

    @AutoWired、@Resource、@Inject

    都可以给属性注入值
    1、@AutoWired

    • 默认优先按照类型去容器中找对应的组件,找到就赋值。

    • 如果该类型的组件有多个,再将属性名作为组件的id去容器中查找

    • required属性:当required=false时,表示这个组件不是必须的

    • @Qualifier需要和@AutoWired一起使用,用于指定id

    • @Primary指定当前bean是首选的

    • 可以在构造器、参数、方法、属性上标注

      • 我们可以在有参构造函数上标注@Autowired,让IOC容器创建组件时调用该类的有参构造方法

        • 默认加在IOC容器中的组件,容器启动会调用无参构造器创建对象,再进行初始化赋值等操作;
        • 如果组件只有一个有参构造器,这个有参构造器的@Autowired可以省略
      • 标在参数上,自定义类型的值从IOC容器中获取

      • 标注在方法上,Spring容器创建当前对象,就会调用方法,完成赋值

      • 标在属性上,会从ioc容器中获取并赋值

    2、@Resource

    • 可以和@Autowired一样实现自动装配功能,但默认是按组件名称进行装配
    • 可以给属性name赋值,自定义组件名称

    3、@Inject

    • 需要导入javax.inject依赖

    • 和@Autowired功能一样,但没有required=false的功能

    使用Spring容器底层组件(实现xxxAware)

    自定义组件想要使用Spring容器底层的一些组件(ApplicationContext、BeanFactory),需要自定义组件实现xxxAware接口

    在创建对象的时候,会调用接口规定的方法,注入相关的组件

    例如:在这里插入图片描述
    自定义类实现ServletContextAware接口,于是我就能拿到ServletContext进行操作

    @Profile

    1、@Profile注解是Spring为我们提供的可以根据当前环境,动态的激活和切换一系列组件的功能

    • 可以写在类上和方法上
    • 例如:@Profile(“test”),只有在运行环境时test的时候,组件才会生效
    • 没有添加@Profile注解,表示任何环境下都生效

    2、指定运行环境的方式:

    • 命令行方式,启动时指定参数:-Dspring.profiles.active=test

    • 代码方式步骤:

      • 1、使用无参构造器创建application,AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext();
      • 2、设置需要激活的环境:applicationContext.getEnvironment().setActiveProfiles(“test”, “dev”);
      • 3、注册主配置类:applicationContext.register(MainConfig11.class);
      • 4、刷新容器:applicationContext.refresh();

    SpringAOP

    • 【动态代理】
    • 指在程序运行期间动态将某段代码切入到指定方法的指定玩位置进行运行的编程方式
      在这里插入图片描述

    步骤

    使用步骤

    • 导入aop模块;Spring AOP:(spring-aspects)

      <dependency>
          <groupId>org.springframeworkgroupId>
          <artifactId>spring-aspectsartifactId>
      dependency>
      
      • 1
      • 2
      • 3
      • 4
    • 业务逻辑类

    • 切面类

    • 给切面类的目标方法标注解

      • 前置通知(@Before):在目标方法运行之前运行
      • 后置通知(@After):在目标方法运行结束之后运行(无论方法正常结束还是异常结束)
      • 返回通知(@AfterReturning):在目标方法正常返回之后运行
      • 异常通知(@AfterThrowing):在目标方法出现异常以后运行
      • 环绕通知(@Around):动态代理,手动推进目标方法运行(joinPoint.procced())
    • 将切面类和业务逻辑类(目标方法所在类)都`加入到容器中

    • 必须告诉Spring哪个类是切面类(给切面类上加一个注解:@Aspect)

    • 给配置类中加 @EnableAspectJAutoProxy 【开启基于注解的aop模式】.

    注意:JoinPoint 一定要作为第一个参数

    例子

    • 1、引入依赖
    • 2、业务逻辑类,例如定义一个controller:
      在这里插入图片描述
    • 3、定义切面类、标注解、声明这是切面类、开启注解对aop的支持,例如:
      在这里插入图片描述

    execution表达式

    参考于:

    表达式的结构

    • 访问修饰符[可省] 返回值 包名.包名.包名...类名.方法名(参数列表)

    其他说明

    • 返回值可以使用通配符 * ,表示任意返回值

    • 包名可以使用通配符 * ,表示任意包。但是有几级包,就需要写几个*

    • 是包和包、包和类等之间的连接符

    • 包名可以使用 … 表示当前包及其子包

    • 类名和方法名都可以使用*来实现通配

    • 参数列表:

      • 可以直接写数据类型:
        • 基本类型直接写名称,例如:int
        • 引用类型写包名.类名的方式,例如:java.lang.String
      • 可以使用通配符 * 表示任意类型,但是必须有参数
      • 可以使用… 表示有无参数均可,有参数可以是任意类型
    • 全通配写法:(表示匹配所有包下的所有类的所有方法): * *..*.*(..)

    • 通常写法:
      切到业务层实现类下的所有方法:* com.ljy.service.impl.*.*(..)

    AnnotationAwareAspectJAutoProxyCreator流程图

    本节图片来源:https://blog.csdn.net/xjhqre/article/details/123264069

    注册AnnotationAwareAspectJAutoProxyCreator的后置处理器流程图:
    在这里插入图片描述

    AnnotationAwareAspectJAutoProxyCreator执行时机流程图:
    在这里插入图片描述
    AnnotationAwareAspectJAutoProxyCreator配置代理流程图:
    在这里插入图片描述

    声明式事务 (略)

    视频地址:https://www.bilibili.com/video/BV1gW411W7wy?p=38

    基本原理:

    • 执行目标方法,如果异常,获取到事务管理器,利用事务管理回滚操作;
    • 如果正常,利用事务管理器,提交事务

    底层类

    1、BeanPostProcessor

    bean后置处理器,bean创建对象初始化前后进行拦截工作的;主要有以下两个方法:

    • postProcessBeforeInitialization,在bean初始化之前执行
    • postProcessAfterInitialization,在bean初始化之后执行

    2、BeanFactoryPostProcessor

    beanFactory的后置处理器:

    • 在BeanFactory标准初始化之后调用,来定制和修改BeanFactory的内容
    • 所有的bean定义已经保存加载到beanFactory,但是bean的实例还未创建

    3、BeanDefinitionRegistryPostProcessor

    它extends BeanFactoryPostProcessor

    • 其优先于BeanFactoryPostProcessor执行;在所有bean定义信息将要被加载,bean实例还未创建的时候执行

    • 可以利用BeanDefinitionRegistryPostProcessor给容器中再额外添加一些组件

    4、ApplicationListener

    ApplicationListener extends EventListener

    • 该类有一个方法:onApplicationEvent,当事件发生后执行
    • 可以监听自定义事件,需要自定义事件,然后可调用applicationContext.publishEvent()发布事件

    自定义事件示例:

    • 定义事件,继承ApplicationEvent
      在这里插入图片描述

    • 定义监听器,监听自定义事件
      在这里插入图片描述

    • 发布自定义事件:
      在这里插入图片描述

      • 另外,个人觉得可以把run作设置为静态成员变量,以便其他地方发布事件,或者实现ApplicationContextAware接口,也可以获得ApplicationContext
    • 结果:在这里插入图片描述

  • 相关阅读:
    Java中的ORM框架——myBatis
    爬虫 — JsonPath 和 CSV 文件读写
    JScript的编写、调试
    【系统性学习】Linux Shell易忘重点整理
    java计算机毕业设计全国消费水平展示平台源码+数据库+系统+lw文档+mybatis+运行部署
    Mysql数据库基础知识总结,结构分明,内容详细
    PLSQL数据库Mybatis学习Day02
    ArcgisForJS如何使用ArcGIS Server的缓冲区几何服务?
    ubuntu18.04 ros 安装 gazebo9
    HotSpot垃圾收集算法实现细节
  • 原文地址:https://blog.csdn.net/m0_55155505/article/details/125840428