获取接口所有实现类的两种方法 spring和java spi

  • Post author:
  • Post category:java


有时候,根据业务逻辑的需求,我们想要获取到某个接口的所有实现类。在这里大致介绍两种方式:


1.借助Spring容器实现

Spring作为一个容器,管理着一个项目中所有经过配置的Java类(xml配置文件或Annotation方式)。如果某个接口的所有实现类均被Spring托管了,那么通过Spring就可以很简单的返回这些实现类。

import org.springframework.beans.BeansException;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.stereotype.Component;
 
@Component
public class ServiceLocator implements ApplicationContextAware{
    /**
     * 用于保存接口实现类名及对应的类
     */
    private Map<String, IService> map;
 
    /**
     * 获取应用上下文并获取相应的接口实现类
     * @param applicationContext
     * @throws BeansException
     */
    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        //根据接口类型返回相应的所有bean
        Map<String, IService> map = applicationContext.getBeansOfType(IService.class);
    }
 
    public Map<String, IService> getMap() {
        return map;
    }
}

当然, 直接适用spring的@Autowired注解实现也是可以的

import org.springframework.beans.BeansException;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.stereotype.Component;
 
@Component
public class ServiceLocator{
    /**
     * 用于保存接口实现类名及对应的类
     */
    @Autowired
    private Map<String, IService> map;
 
    public Map<String, IService> getMap() {
        return map;
    }
}


2.借助ServiceLoader类

ServiceLoader是JDK自带的一个类加载器,位于java.util包当中,作为

A simple service-provider loading facility.

关于SPI知识可以参考网友博客

:


jdk和dubbo的SPI机制

具体使用方式如下:

1.在META-INF/services/目录下用你的接口全路径名称命名一个文件(不加后缀),然后在该文件中一行一个添加你的接口实现类的全路径名。

2.通过load方法来加载出所有的接口实现类

ServiceLoader<MyInterface> loader = ServiceLoader.load(MyInterface.class);

在这里load方法的返回值是一个迭代器,用这个迭代器可以遍历出所有的接口实现类。


总结

以上两种方式,实现的功能都是一样的,实现方式不同,底层用的技术一样的,都是反射。至于选择哪一种,我建议如果项目中的接口实现类都被Spring托管了,那当然是直接用Spring了。如果没有用到Spring的话,那就用ServiceLoader,这个肯定是没有问题的。