- 原文地址:Java Service Loader vs. Spring Factories Loader
- 原文做者:Nicolas Frankel
- 译文出自:掘金翻译计划
- 本文永久连接:github.com/xitu/gold-m…
- 译者:HearFishle
- 校对者:Endone,ziyin feng
IoC 并不只限于解决模块内类与类之间的依赖耦合问题,其一样适用于模块与模块之间。OSGi 一直致力于这方面的工做。但其实 Java 和 Spring 都提供了对 IoC 的支持。html
Java 自己提供了一种很简便的方式来支持 IoC,它经过使用 [Service Loader] (https://docs.oracle.com/javase/6/docs/api/java/util/ServiceLoader.html) 来实现,其能够获取到工程类路径内指定接口的实现类。这使咱们能够在运行期间获知类路径内包含哪些可用的实现类,从而作到接口定义和多个实现模块(JAR 包)之间的依赖解耦。前端
SLF4J 做为一个日志框架正是使用了这个方法。SLF4J 自己只提供日志操做接口,其余的日志系统基于这些接口进行实现(如 Logback 和 Log4J 等)。用户只需经过调用 SLF4J 的接口来记录日志,而具体的实现则交由工程类路径中可用的实现类来执行。java
为了使用 Service Loader,首先须要在类所在工程的类路径下面创建 'META-INF/services' 目录,而后根据接口名在该目录建立一个文件。该文件的文件名必须是接口的彻底限定名,其内容是可用实现的限定名列表。例如,对于 ch.frankel.blog.serviceloader.Foo
这个接口,文件名应该是 META-INF/services/ch.frankel.blog.serviceloader.Foo
,文件的内容多是以下这样的:android
ch.frankel.blog.serviceloader.FooImpl1
ch.frankel.blog.serviceloader.FooImpl2
复制代码
其中包含的类必须实现 ch.frankel.blog.serviceloader.Foo
接口。ios
使用 Service Loader 获取实现类的代码很是简单:git
ServiceLoader<Foo> loader = ServiceLoader.load(Foo.class);
loader.iterator();
复制代码
核心的 Spring 库以工厂模式集成了 Java 的 Service Loader。例如,下面的代码假定工程内至少有一个可选的 Foo 接口的实现类:github
@Configuration
public class ServiceConfiguration {
@Bean
public ServiceListFactoryBean serviceListFactoryBean() {
ServiceListFactoryBean serviceListFactoryBean = new ServiceListFactoryBean();
serviceListFactoryBean.setServiceType(Foo.class);
return serviceListFactoryBean;
}
}
Object object = serviceListFactoryBean.getObject();
复制代码
很明显,从调用返回来看,须要进一步操做才能获得正确格式的数据(注意:serviceListFactoryBean 是一个链表)。spring
除了集成 Java 的 Service Loader 以外,Spring 还提供了另外一种 IoC 的实现。其只须要添加一个简单的配置文件,文件名必须为 spring.factories
而且放到 META-INF
下。从代码的角度看,这个文件经过静态方法 SpringFactoriesLoader.loadFactories()
来读取。Spring 的这个实现确实让你吃惊。 调用的代码不能再简单了:后端
List<Foo> foos = SpringFactoriesLoader.loadFactories(Foo.class, null);
复制代码
上面第二个可选参数是类加载器api
相对于 Java Service Loader,主要有两方面的区别:
spring.factories
中并无要求键是一个接口而且实现它的值。例如,Spring Boot 使用这种方法来初始化类实例:配置中键内容为一个注解,如 org.springframework.boot.autoconfigure.EnableAutoConfiguration
,而值是则能够是标注了 @Configuration
注解的类。若是灵活使用,能够去完成更多更复杂的设计。这篇文章的资源能够在 GitHub 的 Maven 格式下找到。
延伸阅读:
若是发现译文存在错误或其余须要改进的地方,欢迎到 掘金翻译计划 对译文进行修改并 PR,也可得到相应奖励积分。文章开头的 本文永久连接 即为本文在 GitHub 上的 MarkDown 连接。
掘金翻译计划 是一个翻译优质互联网技术文章的社区,文章来源为 掘金 上的英文分享文章。内容覆盖 Android、iOS、前端、后端、区块链、产品、设计、人工智能等领域,想要查看更多优质译文请持续关注 掘金翻译计划、官方微博、知乎专栏。