【Spring Boot系列4】spring.factories配置_sprint.factories定义多个配置_楼仔的博客-程序员宅基地

技术标签: spring boot  SpringBoot  

往期精选(欢迎转发~~)

​结合具体实例,讲解SpringFactories机制。

问题引入

未注入报错

昨天开发项目时,有两个独立的Module,我需要在Module A中,调用Module B的方法:

@Slf4j
@SpringBootTest(classes = Application.class)
@RunWith(SpringJUnit4ClassRunner.class)
public class AlertMsgTest {
    // Module B的类
    @Autowired
    private AlarmMsgMqConsumer alarmMsgMqConsumer;
    @Test
    public void testLoad() throws MQClientException {
        alarmMsgMqConsumer.init();
        return;
    }
}

调用过程中提示报错:

我们看看AlarmMsgMqConsumer类:

@Component
public class AlarmMsgMqConsumer {
}

可以发现,报错的原因是因为Module A没有将AlarmMsgMqConsumer加入到Spring Bean中,那么我们直接在Module A中通过@ComponentScan去加载Module B的类么?这样其实不能实现解耦,这时可以引入spring.factories。

自动加载

为了能让Module B中的Bean能提前注入到Spring中,我们新增类:

@Configuration
@ComponentScan(basePackages = "com.mi.info.sales.middle.alarm")
public class AlertAutoConfig {
}

其中“com.mi.info.sales.middle.alarm”就是Module B的路径,那怎样将这文件能自动启动呢,我们再新增一个spring.factories文件:

org.springframework.boot.autoconfigure.EnableAutoConfiguration=\
com.mi.info.sales.middle.alarm.AlertAutoConfig

通过spring.factories,将AlertAutoConfig配置进去,可以实现自动加载的功能,下面是项目目录结构:

SpringFactories机制

Spring Boot中有一种非常解耦的扩展机制:Spring Factories。这种扩展机制实际上是仿照Java中的SPI扩展机制来实现的。

什么是 SPI机制

SPI的全名为Service Provider Interface.大多数开发人员可能不熟悉,因为这个是针对厂商或者插件的。在java.util.ServiceLoader的文档里有比较详细的介绍。

简单的总结下java SPI机制的思想。我们系统里抽象的各个模块,往往有很多不同的实现方案,比如日志模块的方案,xml解析模块、jdbc模块的方案等。面向的对象的设计里,我们一般推荐模块之间基于接口编程,模块之间不对实现类进行硬编码。一旦代码里涉及具体的实现类,就违反了可拔插的原则,如果需要替换一种实现,就需要修改代码。为了实现在模块装配的时候能不在程序里动态指明,这就需要一种服务发现机制。

java SPI就是提供这样的一个机制:为某个接口寻找服务实现的机制。有点类似IOC的思想,就是将装配的控制权移到程序之外,在模块化设计中这个机制尤其重要。

Spring Boot中的SPI机制

在Spring中也有一种类似与Java SPI的加载机制。它在META-INF/spring.factories文件中配置接口的实现类名称,然后在程序中读取这些配置文件并实例化。这种自定义的SPI机制是Spring Boot Starter实现的基础。

在日常工作中,我们可能需要实现一些SDK或者Spring Boot Starter给被人使用,这个使用我们就可以使用Factories机制。Factories机制可以让SDK或者Starter的使用只需要很少或者不需要进行配置,只需要在服务中引入我们的jar包。

spring.factories

spring.factories的是通过Properties解析得到的,所以我们在写文件中的内容都是按照下面这种方式配置的:

com.xxx.interface=com.xxx.classname

比如spring-boot包中的spring.factories文件:

# PropertySource Loaders
org.springframework.boot.env.PropertySourceLoader=\
org.springframework.boot.env.PropertiesPropertySourceLoader,\
org.springframework.boot.env.YamlPropertySourceLoader

# Run Listeners
org.springframework.boot.SpringApplicationRunListener=\
org.springframework.boot.context.event.EventPublishingRunListener

# Error Reporters
org.springframework.boot.SpringBootExceptionReporter=\
org.springframework.boot.diagnostics.FailureAnalyzers

# Application Context Initializers
org.springframework.context.ApplicationContextInitializer=\
org.springframework.boot.context.ConfigurationWarningsApplicationContextInitializer,\
org.springframework.boot.context.ContextIdApplicationContextInitializer,\
org.springframework.boot.context.config.DelegatingApplicationContextInitializer,\
org.springframework.boot.web.context.ServerPortInfoApplicationContextInitializer

# 省略...

SpringFactoriesLoader

在这个类中主要的方法:loadFactories:根据接口类获取其实现类的实例,这个方法返回的是对象列表。

public static <T> List<T> loadFactories(Class<T> factoryClass, @Nullable ClassLoader classLoader) {
    Assert.notNull(factoryClass, "'factoryClass' must not be null");
   
    ClassLoader classLoaderToUse = classLoader;
    if (classLoaderToUse == null) {
        classLoaderToUse = SpringFactoriesLoader.class.getClassLoader();
    }
    // 调用loadFactoryNames获取接口的实现类
    List<String> factoryNames = loadFactoryNames(factoryClass, classLoaderToUse);
    if (logger.isTraceEnabled()) {
        logger.trace("Loaded [" + factoryClass.getName() + "] names: " + factoryNames);
    }
    // 遍历 factoryNames 数组,创建实现类的对象
    List<T> result = new ArrayList<>(factoryNames.size());
    for (String factoryName : factoryNames) {
        result.add(instantiateFactory(factoryName, factoryClass, classLoaderToUse));
    }
    // 排序
    AnnotationAwareOrderComparator.sort(result);
    return result;
}

loadFactoryNames:根据接口获取其接口类的名称,这个方法返回的是类名的列表。

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);
    }
}

instantiateFactory:根据类创建实例对象。

private static <T> T instantiateFactory(String instanceClassName, Class<T> factoryClass, ClassLoader classLoader) {
    try {
        
        Class<?> instanceClass = ClassUtils.forName(instanceClassName, classLoader);
        // 是否实现了指定接口
        if (!factoryClass.isAssignableFrom(instanceClass)) {
            throw new IllegalArgumentException("Class [" + instanceClassName + "] is not assignable to [" + factoryClass.getName() + "]");
        }
        // 创建对象
        return (T) ReflectionUtils.accessibleConstructor(instanceClass).newInstance();
    } catch (Throwable ex) {
        throw new IllegalArgumentException("Unable to instantiate factory class: " + factoryClass.getName(), ex);
    }
}

欢迎大家多多点赞,更多文章,请关注微信公众号“楼仔进阶之路”,点关注,不迷路~~

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/lml200701158/article/details/119215696

智能推荐

SpringBoot整合JPA 完成多对多关联添加查询_jpa关联查询添加状态查询_小超zzzzzzz的博客-程序员宅基地

SpringBoot整合JPA 完成关联查询以专家和事件为例sql语句CREATE TABLE `expert` ( `id` int(11) NOT NULL AUTO_INCREMENT, `expertName` varchar(255) DEFAULT NULL COMMENT '专家名称', PRIMARY KEY (`id`)) ENGINE=InnoDB AUTO_INCREMENT=6 DEFAULT CHARSET=utf8;CREATE TABLE `eve

XYNU 1249: 均分纸牌(playcard)_DrTimer的博客-程序员宅基地

1249: 均分纸牌(playcard)时间限制: 1 Sec内存限制: 128 MB提交: 141解决: 81您该题的状态:已完成[提交][状态][讨论版]题目描述有 N 堆纸牌,编号分别为 1,2,…, N。每堆上有若干张,但纸牌总数必为 N 的倍数。可以在任一堆上取若于张纸牌,然后移动。 移牌规则为:在编号为 1 堆上取的纸牌,只能移到编号为 2 的堆上;在编号为 N...

条形码识别_zaitianaoxiang的博客-程序员宅基地

目前没有多余时间研究将开源地址放下,感兴趣的人可以研究一下,多多交流。。。http://sourceforge.net/projects/zbar/files/iPhoneSDK/

Linux 重定向_zeroonetwothree的博客-程序员宅基地

&lt; :输入重定向&gt; :输出重定向&gt;&gt; :输出重定向,进行追加,不会覆盖之前内容&lt;&lt; :标准输入来自命令行的一对分隔号的中间内容.需求:向文件内自动输入一些内容,举一个简单的例子。cat &lt;&lt;EOF&gt; out.txt&gt;This is a sample&gt;second line&gt;EOF通过上面的重定向知识,我们可以知道 cat 命令获取了来自自定义分隔符‘EOF’中的内容作为标准输入,然后标准输出重定位到 out.txt

IOS中Json解析的四种方法_zhengxiuchen86的博客-程序员宅基地

作为一种轻量级的数据交换格式,json正在逐步取代xml,成为网络数据的通用格式。 有的json代码格式比较混乱,可以使用此“http://www.bejson.com/”网站来进行JSON

Docker 从入门到精通(三)一 网络配置_流風餘韻的博客-程序员宅基地

docker 是Linux下面的容器技术,是目前最火的开源技术之一,上次我们了解了docker的本地仓库的搭建,今天我们介绍下docker的容器配置步骤,网络的一些配置。

随便推点

hello world及内核模块 && 把.ko的驱动改生成.o的场景_ko变成.o_zhandoushi1982的博客-程序员宅基地

(1)C普通程序实例启动终端,首先用VI编写一个C程序:vi hello.c#include int main(){ printf("hello world!!!/n"); return 0;}       接着用GCC进行编译:gcc -o hello hello.c,最后运行该程序:./hello,在终端上你会看到:hell

响应式前端框架Bootstrap系列(1)浅谈Bootstrap_黄泽平的博客-程序员宅基地

有一段时间没写博客了,在忙着公司的项目上线。最近终于有了点空闲时间,打算写一些关于前端响应式框架,头脑中第一时间就冒出了Bootstrap,毕竟它是Github上Star数最多的框架之一,下面简称bs。虽然bs的中文资料还蛮多的,百度一下就有一堆,但为了方便记忆,我还是想写一系列博文把它们分类总结一下。平心而论,作为一个前端工程师,我并不会很热衷于使用bs框架,因为我参与的项目中至少会

智慧背囊_紫颖的博客-程序员宅基地

今天在逛论坛发现了以下小故事,好多以前都听过,但是觉得有寓意,就拿来和大家分享~1.甲去买烟,烟29元,但他没火柴,跟店员说:“顺便送一盒火柴吧。”店员没给。  乙去买烟,烟29元,他也没火柴,跟店员说:“便宜一毛吧。”最后,他用这一毛买一盒火柴。  这是最简

利用shell脚本重启node.js_shell脚本启动node.js_zhangfei8625的博客-程序员宅基地

利用shell脚本管理node.js启动相关动作利用shell脚本管理node.js程序的启动,停止和重启动作启动node.js入口文件停止node.js入口文件重启node.js入口文件快捷键脚本名 start 脚本名 stop 脚本名 reload代码使用shell管理node进程 例如start.sh:#!/bin/bashNODE=`which node`