【Spring注解驱动开发】深入理解Spring的ImportSelector接口

int order() default Ordered.LOWEST_PRECEDENCE;

}

此注解是开启声明式事务的注解,那么它的@Import所导入的类为TransactionManagementConfigurationSelector,那么我们看一下其类图:

在这里插入图片描述

由此可知该类实现类ImportSelector接口。

前面说过,在SpringBoot的自动化配置和@EnableXXX(功能性注解)都有ImportSelector接口的存在,那我们就来自己定义一个@EnableXXX注解来更加深刻的理解ImportSelector接口。

自定义@EnableXXX注解


在这里我们先准备两个Spring的项目工程:spring-project与ssm-project,其中spring-project里我们先创建好如下结构目录:

在这里插入图片描述

创建实体类

package org.hzgj.spring.study.bean

public class StudentBean{

private Integer id;

private String name;

//省略setter和gettter

}

创建ImportSelector接口的实现类

package org.hzgj.spring.study.config;

import org.springframework.beans.BeansException;

import org.springframework.beans.factory.BeanFactory;

import org.springframework.beans.factory.BeanFactoryAware;

import org.springframework.context.annotation.ImportSelector;

import org.springframework.core.type.AnnotationMetadata;

public class SpringStudySelector implements ImportSelector, BeanFactoryAware {

private BeanFactory beanFactory;

@Override

public String[] selectImports(AnnotationMetadata importingClassMetadata) {

importingClassMetadata.getAnnotationTypes().forEach(System.out::println);

System.out.println(beanFactory);

return new String[]{AppConfig.class.getName()};

}

@Override

public void setBeanFactory(BeanFactory beanFactory) throws BeansException {

this.beanFactory = beanFactory;

}

}

在这里我们实现ImportSelector接口和BeanFactoryAware接口,重写selectImports方法,最后我们返回的是AppConfig的类名,同时打印出相关的注解元数据与BeanFactory

自定义@EnableSpringStudy注解

package org.hzgj.spring.study.annotation;

import org.hzgj.spring.study.config.SpringStudySelector;

import org.springframework.context.annotation.Import;

import java.lang.annotation.*;

@Retention(RetentionPolicy.RUNTIME)

@Documented

@Target(ElementType.TYPE)

@Import(SpringStudySelector.class)

public @interface EnableSpringStudy {

}

在这里我们仿照@EnableTransactionManagement来实现自定义注解,注意使用@Import导入我们刚才写的SpringStudySelector。

创建配置类

package org.hzgj.spring.study.config;

import org.hzgj.spring.study.bean.StudentBean;

import org.springframework.context.annotation.Bean;

import org.springframework.context.annotation.Configuration;

@Configuration

public class AppConfig {

@Bean

public StudentBean studentBean() {

StudentBean studentBean = new StudentBean();

studentBean.setId(19);

studentBean.setName(“admin”);

return studentBean;

}

}

当都完成以后我们打个jar包,准备引入至其他工程:

在这里插入图片描述

使用自定义@EnableXXX注解


完成ssm-project工程中的AppConfig配置类

  1. 首先我们将刚才的spring.jar导入到ssm-project工程里

  2. 在对应的配置类上添加上spring-project中定义的@EnableSpringStudy注解

@Configuration //表明此类是配置类

@ComponentScan // 扫描自定义的组件(repository service component controller)

@PropertySource(“classpath:application.properties”) // 读取application.properties

@MapperScan(“com.bdqn.lyrk.ssm.study.app.mapper”) //扫描Mybatis的Mapper接口

@EnableTransactionManagement //开启事务管理

@EnableSpringStudy

public class AppConfig {

//…省略配置代码

}

3)编写Main方法

public static void main(String[] args) throws IOException {

AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(AppConfig.class);

StudentBean studentBean = applicationContext.getBean(StudentBean.class);

System.out.println(studentBean.getName());

}

运行后输出结果:

org.springframework.context.annotation.Configuration

org.springframework.context.annotation.ComponentScan

org.springframework.context.annotation.PropertySource

org.mybatis.spring.annotation.MapperScan

org.springframework.transaction.annotation.EnableTransactionManagement

org.hzgj.spring.study.annotation.EnableSpringStudy

org.springframework.beans.factory.support.DefaultListableBeanFactory@4b9e13df: defining beans [org.springframework.context.annotation.internalConfigurationAnnotationProcessor,org.springframework.context.annotation.internalAutowiredAnnotationProcessor,org.springframework.context.annotation.internalRequiredAnnotationProcessor,org.springframework.context.annotation.internalCommonAnnotationProcessor,org.springframework.context.event.internalEventListenerProcessor,org.springframework.context.event.internalEventListenerFactory,appConfig,propertiesConfig,logAspect,studentService]; root of factory hierarchy

admin

从这里我们可以看到ImportSelector接口中的方法参数,可以获取ssm-project项目下AppConfig的所有注解,并且能够获取当前BeanFactory所有配置的Bean。

ImportSelector源码分析


这个接口在哪里调用呢?我们可以来看一下ConfigurationClassParser这个类的processImports方法。

private void processImports(ConfigurationClass configClass, SourceClass currentSourceClass,

Collection importCandidates, boolean checkForCircularImports) {

if (importCandidates.isEmpty()) {

return;

}

if (checkForCircularImports && isChainedImportOnStack(configClass)) {

this.problemReporter.error(new CircularImportProblem(configClass, this.importStack));

}

else {

this.importStack.push(configClass);

try {

for (SourceClass candidate : importCandidates) {            //对ImportSelector的处理

if (candidate.isAssignable(ImportSelector.class)) {

// Candidate class is an ImportSelector -> delegate to it to determine imports

Class<?> candidateClass = candidate.loadClass();

ImportSelector selector = BeanUtils.instantiateClass(candidateClass, ImportSelector.class);

ParserStrategyUtils.invokeAwareMethods(

selector, this.environment, this.resourceLoader, this.registry);

if (this.deferredImportSelectors != null && selector instanceof DeferredImportSelector) {                //如果为延迟导入处理则加入集合当中

this.deferredImportSelectors.add(

new DeferredImportSelectorHolder(configClass, (DeferredImportSelector) selector));

}

else {                //根据ImportSelector方法的返回值来进行递归操作

String[] importClassNames = selector.selectImports(currentSourceClass.getMetadata());

Collection importSourceClasses = asSourceClasses(importClassNames);

processImports(configClass, currentSourceClass, importSourceClasses, false);

}

}

else if (candidate.isAssignable(ImportBeanDefinitionRegistrar.class)) {

// Candidate class is an ImportBeanDefinitionRegistrar ->

// delegate to it to register additional bean definitions

Class<?> candidateClass = candidate.loadClass();

ImportBeanDefinitionRegistrar registrar =

BeanUtils.instantiateClass(candidateClass, ImportBeanDefinitionRegistrar.class);

ParserStrategyUtils.invokeAwareMethods(

registrar, this.environment, this.resourceLoader, this.registry);

configClass.addImportBeanDefinitionRegistrar(registrar, currentSourceClass.getMetadata());

}

else {              // 如果当前的类既不是ImportSelector也不是ImportBeanDefinitionRegistar就进行@Configuration的解析处理

// Candidate class not an ImportSelector or ImportBeanDefinitionRegistrar ->

// process it as an @Configuration class

this.importStack.registerImport(

currentSourceClass.getMetadata(), candidate.getMetadata().getClassName());

processConfigurationClass(candidate.asConfigClass(configClass));

}

}

}

catch (BeanDefinitionStoreException ex) {

throw ex;

}

catch (Throwable ex) {

throw new BeanDefinitionStoreException(

“Failed to process import candidates for configuration class [” +

configClass.getMetadata().getClassName() + “]”, ex);

}

finally {

this.importStack.pop();

}

}

}

在这里我们可以看到ImportSelector接口的返回值会递归进行解析,把解析到的类全名按照@Configuration进行处理。

最后

由于篇幅原因,就不多做展示了
etMetadata().getClassName() + “]”, ex);

}

finally {

this.importStack.pop();

}

}

}

在这里我们可以看到ImportSelector接口的返回值会递归进行解析,把解析到的类全名按照@Configuration进行处理。

最后

[外链图片转存中…(img-qGeQuDqs-1718783762813)]

[外链图片转存中…(img-Fz6PZ0DK-1718783762814)]

[外链图片转存中…(img-uRBv4dty-1718783762815)]

[外链图片转存中…(img-sf96YimS-1718783762815)]

[外链图片转存中…(img-42XsoWGq-1718783762816)]

[外链图片转存中…(img-N4rPcDHc-1718783762816)]

由于篇幅原因,就不多做展示了

  • 5
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值