如何让Spring Boot自动配置为我所用

一句话概况看完本文你将有的收获

  • 掌握 Spring Boot 自动配置的基本操作与原理,学会如何开发被 spring 容器管理的开发工具包(SDK)。

带着问题找答案

  • 众所周知,自定义类被 Spring 管理的两个基本条件是:
  1. 类被@Service等注解标注
  2. 类被 Spring 容器扫描
  • 在 Spring Boot 体系下开发,我们都会将需要 Spring 管理的对象进行注解标注,并且我们知道 Spring Boot 中默认扫描路径是 Spring Boot 主应用所在包及其子包下。
  • Spring Boot 的扫描规则在我们的项目中很好用,但是我们现在想开发一个被 Spring 管理的 SDK 该怎么办呢?显然限制 SDK 的包名和指定扫描路径都不是最优解,都会增加项目的负担和耦合。

如何解决问题?

  • 显然这个需求与 Spring Cloud 中一些组件的使用是一致的,于是很容易就能想到解决问题的办法肯定在 Spring Boot 的自动配置上,下文将分为两部分讲解,如何使用自动配置来开发 SDK(知其然)和原理解析(知其所以然)

知其然篇

举个需求栗子:)

  • 想为应用服务来监控每个请求(请求地址、状态、耗时)
  • 请求数据记录到数据库
  • 需要组件化,有多个应用服务有同样需求
  • 要求尽量让应用服务使用起来简单方便

HOW TO DO?

  1. 开发一个 SDK,其中有Filter并且具备操作数据库的能力 --> 在应用服务中开发,这两个功能再简单不过了,那么问题来了,在 SDK 中开发,如何才能让 spring 容器接管 SqlSession 和业务处理 Service 层呢?
  2. 使用 Spring Boot 自动配置功能加载需要被 Spring 容器管理的类

主要类对象简介

  1. class MetricsService --> Service 层,处理逻辑及 DAO
  2. interface MetricsHttpRequestMapper --> Spring + Mybatis 的 mapper,相当于 DAO 层
  3. class MetricsHttpRequestContainerFilter implements ContainerRequestFilter, ContainerResponseFilter --> Jersey框架中的自定义Filter,具有请求前过滤和请求后过滤作用。如果没用过Jersey的不用太在意这个类的使用,毕竟这只是个举个栗子的需求,如果真的有这个需求的同学,可以用各自框架的Filter定义来替换,就想知道这个Filter更多细节的同学可以私聊,这里不展开说了。
  4. class BeanConfiguration --> SDK 中 Bean对象定义类,建议将需要 Spring 管理的类分类到此种类中控制定义,等会讲清楚此类的使用就知其然了
  5. class SqlSessionFactoryConfig --> Spring + Mybatis 定义SqlSessionFactoryBean的类,在此文中此类不展开说了

部分代码展示说明

1. Service关键代码展示
public class MetricsServiceImpl implements MetricsService{

    // 此处说明该类需要被 Spring 容器管理
    @Autowired
    MetricsHttpRequestMapper httpRequestMapper;

    /**
     * @Param Filter解析请求后的需要处理对象
     */
    @Override
    public void addHttpRequest(MetricsHttpRequest model) {
        // 处理逻辑
    }
}
2. BeanConfiguration关键代码展示
public class BeanConfiguration {

    // 建议Service层Bean分类后统一交由此种类配置
    @Bean
    public MetricsService metricsService(){
        return new MetricsServiceImpl();
    }
}
3. 如何加载 BeanConfiguration 关键部分
3.1 resources 文件夹下建立该文件 META-INF/spring.factories
3.2 内容:
org.springframework.boot.autoconfigure.EnableAutoConfiguration=\
com.需要自动配置的类1,\
需要自动配置的类包名.BeanConfiguration

4. 结构如图所示

SDK自动配置文件展示.png

知其然总结

  • 配置完成后,应用主类只要开启自动配置就可以完成 SDK 中类被 Spring 容器管理的目标了
  • 关键部分还是在 spring.factories 文件的定义上,只要了解到这点,使用已经不是问题了。
  • 下部分将简单讲述原理,有兴趣的可以继续

知其所以然篇

如何寻找答案

  • 自动配置的答案肯定要在 Spring Boot 如何实现自动配置的源码里找答案啦:)

源码分析

  1. 注解开始: @EnableAutoConfiguration 为开启 Spring Boot 自动配置的注解
  2. 查看该注解:
//省略部分代码
@AutoConfigurationPackage
@Import(EnableAutoConfigurationImportSelector.class)
public @interface EnableAutoConfiguration {
//省略部分代码
}
  1. 注解会加载EnableAutoConfigurationImportSelector
public class EnableAutoConfigurationImportSelector
        implements DeferredImportSelector, BeanClassLoaderAware, ResourceLoaderAware,
        BeanFactoryAware, EnvironmentAware, Ordered {
//省略部分代码
    @Override
    public String[] selectImports(AnnotationMetadata metadata) {
        if (!isEnabled(metadata)) {
            return NO_IMPORTS;
        }
        try {
            AnnotationAttributes attributes = getAttributes(metadata);
            List<String> configurations = getCandidateConfigurations(metadata,
                    attributes);
            configurations = removeDuplicates(configurations);
            Set<String> exclusions = getExclusions(metadata, attributes);
            configurations.removeAll(exclusions);
            configurations = sort(configurations);
            recordWithConditionEvaluationReport(configurations, exclusions);
            return configurations.toArray(new String[configurations.size()]);
        }
        catch (IOException ex) {
            throw new IllegalStateException(ex);
        }
    }

    //获取META-INF/spring.factories文件中定义类的方法
    protected List<String> getCandidateConfigurations(AnnotationMetadata metadata,
            AnnotationAttributes attributes) {
        List<String> configurations = SpringFactoriesLoader.loadFactoryNames(
                getSpringFactoriesLoaderFactoryClass(), getBeanClassLoader());
        Assert.notEmpty(configurations,
                "No auto configuration classes found in META-INF/spring.factories. If you "
                        + "are using a custom packaging, make sure that file is correct.");
        return configurations;
    }
}

该类通过 SpringFactoriesLoader.loadFactoryNames 获取到 META-INF/spring.factories 中定义的类,组成 List<String>
传递给 Spring 处理

  1. 可以看到是在 SpringFactoriesLoader 解析 META-INF/spring.factories 类的
public abstract class SpringFactoriesLoader {

    public static final String FACTORIES_RESOURCE_LOCATION = "META-INF/spring.factories";
    //省略部分代码
    public static List<String> loadFactoryNames(Class<?> factoryClass, ClassLoader classLoader) {
        String factoryClassName = factoryClass.getName();
        try {
            Enumeration<URL> urls = (classLoader != null ? classLoader.getResources(FACTORIES_RESOURCE_LOCATION) :
                    ClassLoader.getSystemResources(FACTORIES_RESOURCE_LOCATION));
            List<String> result = new ArrayList<String>();
            while (urls.hasMoreElements()) {
                URL url = urls.nextElement();
                Properties properties = PropertiesLoaderUtils.loadProperties(new UrlResource(url));
                String factoryClassNames = properties.getProperty(factoryClassName);
                result.addAll(Arrays.asList(StringUtils.commaDelimitedListToStringArray(factoryClassNames)));
            }
            return result;
        }
        catch (IOException ex) {
            throw new IllegalArgumentException("Unable to load [" + factoryClass.getName() +
                    "] factories from location [" + FACTORIES_RESOURCE_LOCATION + "]", ex);
        }
    }
    //省略部分代码
}

该类展示了为什么是从 resource文件夹 中的 META-INF/spring.factories 文件中解析

总结

  • 通过此文可以基本掌握自动配置是如何运作的,其中还有些细节在这里就没有展开说了,比如 @Bean , @ConditionalOnClass 介绍这种注解的文章很多很多,就不重复说了。
  • 文中我举的需求例子不是随意举的,在此想向广大同学们询问一下有没有较好的处理该需求的开源组件,如有非常希望能够告知。在此谢谢大家!

参考资料:
https://docs.spring.io/spring-boot/docs/current/reference/html/boot-features-developing-auto-configuration.html

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,456评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,370评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,337评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,583评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,596评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,572评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,936评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,595评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,850评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,601评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,685评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,371评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,951评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,934评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,167评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,636评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,411评论 2 342

推荐阅读更多精彩内容