SpringBoot自定义starter增强版(二)——扫描自定义注解

1. 情况说明

上篇文章写了一个简单版的SpringBoot自定义starter。其核心原理就是SpringBoot在启动的过程中,会拿到类路径下所有jar包中的META-INF/spring.factories文件,拿到里面的org.springframework.boot.autoconfigure.EnableAutoConfiguration配置项,来判断配置的类是否满足自动装配的条件。

如果自动配置类生效了,就可以做很多事情了。很多第三方jar包的集成就是基于这个点,比如MybatisSpring的集成。

下面就模拟一个场景,自己自定义一个注解@Scorpios,让SpringBoot在启动过程中,把指定注解的类扫描进Spring容器中,模拟第三方应用的扩展。

其实现思路和代码,参考了Spring源码

2. 步骤

  • 引入对应的依赖
  • 编写自定义注解:@Scorpios注解
  • 编写自动配置类:@Configuration注解、@Import注解
  • 编写具体扫描实现类:CustomEnhanceRegister
  • resources/META-INF/spring.factories中配置自定义的自动装配类

本文知识点涉及@Import注解及ImportBeanDefinitionRegistrar接口的使用,在阅读前,可参考以下文章,熟悉其中知识点:

向Spring容器中注册组件的几种方式:

https://blog.csdn.net/zxd1435513775/article/details/100625879

Spring源码系列(八)——Mybatis是如何整合进Spring源码分析

https://blog.csdn.net/zxd1435513775/article/details/121180974

3. 代码实现

目录结构

在这里插入图片描述

3.1 pom文件

最后要把自定义的starter工程打成jar,让其他工程引用

 <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
        </dependency>
    </dependencies>

3.2 自动配置类

对于这个自动配置类,要配置在META-INF/spring.factories文件里。一旦让这个自动配置类生效,就可以做很多事情啦。下一篇加强版就在这地方做文章的。

@Slf4j
@Configuration
@Import(CustomEnhanceRegister.class)
@ConditionalOnClass(CustomEnhanceService.class) // 当类路径下有指定类 改配置才有效
public class CustomEnhanceAutoConfiguration {

}

3.3 自定义注解

@Documented
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface Scorpios {

    boolean registerBean() default false;

}

3.4 CustomEnhanceRegister类

此类完成自定义注解的扫描和注册。

这里实现实现EnvironmentAware接口,可以拿到系统的环境变量信息。

我们在配置文件application.properties中配置的配置项,最后都会放到这个Environment中。可参考下面这篇文章:

环境变量(Env)和系统属性(Property)使用

https://blog.csdn.net/zxd1435513775/article/details/103039651

@Slf4j
public class CustomEnhanceRegister implements ImportBeanDefinitionRegistrar, EnvironmentAware {

    static final String DEFAULT_RESOURCE_PATTERN = "**/*.class";

    // 实现EnvironmentAware接口,可以拿到系统的环境变量信息
    private Environment environment;

    @Override
    public void setEnvironment(Environment environment) {
        this.environment = environment;
    }

    @Override
    public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {

        // 扫描指定包,可以手动传入
        String basePackages = "com.scorpios.customenhance";

        List<Class<?>> candidates = scanPackages(basePackages);
        if (candidates.isEmpty()) {
            log.info("扫描指定包下,未发现符合条件的类", basePackages.toString());
            return;
        }
        // 注册扫描到的类
        registerBeanDefinitions(candidates, registry);
    }


    private List<Class<?>> scanPackages(String basePackages) {
        List<Class<?>> candidates = new ArrayList<Class<?>>();
        try {
            candidates.addAll(findCandidateClasses(basePackages));
        } catch (IOException e) {
            log.error("扫描指定包时出现异常", basePackages);
        }
        return candidates;
    }

    // 将指定包下面的符合条件的类返回
    private List<Class<?>> findCandidateClasses(String basePackage) throws IOException {
        List<Class<?>> candidates = new ArrayList<Class<?>>();
        // classpath*:com/scorpios/**/*.class
        String packageSearchPath = ResourcePatternResolver.CLASSPATH_ALL_URL_PREFIX +
                convertPath(basePackage) + '/' + this.DEFAULT_RESOURCE_PATTERN;
        ResourceLoader resourceLoader = new DefaultResourceLoader();
        MetadataReaderFactory readerFactory = new SimpleMetadataReaderFactory(resourceLoader);
        Resource[] resources = ResourcePatternUtils.getResourcePatternResolver(resourceLoader).getResources(packageSearchPath);
        for (Resource resource : resources) {
            MetadataReader reader = readerFactory.getMetadataReader(resource);
            // 过滤哪些类符合条件,此处判断自定义注解 @Scorpios
            if (match(reader.getClassMetadata())) {
                Class<?> candidateClass = transform(reader.getClassMetadata().getClassName());
                if (candidateClass != null) {
                    candidates.add(candidateClass);
                    log.debug("扫描到符合要求的类:" + candidateClass.getName());
                }
            }
        }
        return candidates;
    }


    private void registerBeanDefinitions(List<Class<?>> internalClasses, BeanDefinitionRegistry registry) {
        for (Class<?> clazz : internalClasses) {
            String beanName = ClassUtils.getShortNameAsProperty(clazz);
            RootBeanDefinition rbd = new RootBeanDefinition(clazz);
            registry.registerBeanDefinition(beanName, rbd);
            if (registerSpringBean(clazz)) {
                registry.registerBeanDefinition(beanName, new RootBeanDefinition(clazz));
            }
        }
    }


    private String convertPath(String path) {
        return StringUtils.replace(path, ".", "/");
    }

    // 根据类名返回Class
    private Class<?> transform(String className) {
        Class<?> clazz = null;
        try {
            clazz = ClassUtils.forName(className, this.getClass().getClassLoader());
        } catch (ClassNotFoundException e) {
            log.info("未找到指定类", className);
        }
        return clazz;
    }


    protected boolean match(ClassMetadata metadata) {
        Class<?> clazz = transformToClass(metadata.getClassName());
        if (clazz == null || !clazz.isAnnotationPresent(Scorpios.class)) {
            return false;
        }
        Scorpios scorpios = clazz.getAnnotation(Scorpios.class);
        if (scorpios.registerBean() && isAnnotatedBySpring(clazz)) {
            throw new IllegalStateException("类{" + clazz.getName() + "}已经标识了Spring组件注解");
        }
        // 过滤抽象类,接口,注解,枚举,内部类及匿名类
        return !metadata.isAbstract() && !clazz.isInterface() && !clazz.isAnnotation() && !clazz.isEnum()
                && !clazz.isMemberClass() && !clazz.getName().contains("$");
    }

    private Class<?> transformToClass(String className) {
        Class<?> clazz = null;
        try {
            clazz = ClassUtils.forName(className, this.getClass().getClassLoader());
        } catch (ClassNotFoundException e) {
            log.info("未找到指定类", className);
        }
        return clazz;
    }


    private boolean isAnnotatedBySpring(Class<?> clazz) {
        return clazz.isAnnotationPresent(Component.class) || clazz.isAnnotationPresent(Configuration.class)
                || clazz.isAnnotationPresent(Service.class) || clazz.isAnnotationPresent(Repository.class)
                || clazz.isAnnotationPresent(Controller.class);
    }

    private boolean registerSpringBean(Class<?> beanClass) {
        return beanClass.getAnnotation(Scorpios.class).registerBean();
    }
}

3.4 具体业务类

使用自定义注解

@Slf4j
@Scorpios
public class CustomEnhanceServiceImpl implements CustomEnhanceService {

    @Override
    public void enhance() {
        log.info("CustomEnhanceServiceImpl...enhance.....");
    }
}

3.5 spring.factories

org.springframework.boot.autoconfigure.EnableAutoConfiguration=com.scorpios.customenhance.autoconfigure.CustomEnhanceAutoConfiguration

4. 测试

新建一个工程,引入上面的jar包,在Controller中,自动注入CustomEnhanceService

@RestController
public class IndexController {

    @Autowired
    CustomEnhanceService customEnhanceService;

    @RequestMapping("/index")
    public String index(){
        customEnhanceService.enhance();
        return "index 8001...";
    }
}

启动日志:

在这里插入图片描述

访问接口地址:http://localhost:8001/index

在这里插入图片描述

自定义注解修饰的CustomEnhanceService已经被调用。

这样是不是就可以集成第三方应用了。。。。。搞定!

具体代码地址如下:

代码地址:https://github.com/Hofanking/springboot-custom-starter-example

  • 1
    点赞
  • 17
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 3
    评论
下面是一个简单的Spring Boot自定义Starter的示例,该Starter实现了一个自定义的HelloWorld功能: 1. 创建Maven项目 首先,我们需要创建一个Maven项目作为我们自定义Starter的项目。在项目的pom.xml中添加Spring Boot的依赖,以及其他需要集成的依赖。 2. 编写自动配置类 在src/main/java目录下创建一个名为HelloWorldAutoConfiguration的类,该类用于自动配置HelloWorld功能: ```java @Configuration @ConditionalOnClass(HelloWorldService.class) @EnableConfigurationProperties(HelloWorldProperties.class) public class HelloWorldAutoConfiguration { @Autowired private HelloWorldProperties properties; @Bean @ConditionalOnMissingBean public HelloWorldService helloWorldService() { HelloWorldService service = new HelloWorldService(); service.setMsg(properties.getMsg()); return service; } @ConfigurationProperties(prefix = "hello.world") public static class HelloWorldProperties { private String msg = "Hello, world!"; public String getMsg() { return msg; } public void setMsg(String msg) { this.msg = msg; } } } ``` 上述代码中,@Configuration注解表示该类是一个配置类,@ConditionalOnClass注解表示只有当HelloWorldService类存在时才进行配置,@EnableConfigurationProperties注解表示将HelloWorldProperties类注入到Spring容器中。在helloWorldService方法中,我们通过读取HelloWorldProperties中的配置来创建一个HelloWorldService实例。 3. 编写Starter类 在src/main/java目录下创建一个名为HelloWorldStarter的类,该类用于将自动配置类注入到Spring容器中: ```java @Configuration @EnableConfigurationProperties(HelloWorldProperties.class) @Import(HelloWorldAutoConfiguration.class) public class HelloWorldStarter { } ``` 上述代码中,@Configuration注解表示该类是一个配置类,@EnableConfigurationProperties注解表示将HelloWorldProperties类注入到Spring容器中,@Import注解表示将HelloWorldAutoConfiguration类注入到Spring容器中。 4. 打包和发布Starter 在命令行中运行以下命令,将自定义Starter打包成jar包: ``` mvn clean package ``` 然后将jar包发布到Maven仓库中。 5. 在项目中使用自定义Starter 在其他Spring Boot项目中的pom.xml文件中添加以下依赖: ```xml <dependency> <groupId>com.example</groupId> <artifactId>hello-world-starter</artifactId> <version>1.0.0</version> </dependency> ``` 在项目中使用以下代码来测试自定义Starter是否生效: ```java @RestController public class TestController { @Autowired private HelloWorldService helloWorldService; @GetMapping("/hello") public String hello() { return helloWorldService.sayHello(); } } ``` 上述代码中,我们通过@Autowired注解注入了HelloWorldService实例,并在hello方法中调用了sayHello方法来测试自定义Starter是否生效。 以上就是Spring Boot自定义Starter的一个简单示例,通过自定义Starter,我们可以将自己的功能快速集成到Spring Boot中,提高开发效率。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

止步前行

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值