Java中的SPI机制及接口多实现调用

1、SPI机制

SPI 全称为 (Service Provider Interface) ,是JDK内置的一种服务提供发现机制。

SPI充分体现了面向接口编程的特点。系统内置接口方法,在实际运行中用户可以自定义实现类来满足不通的实现需求。

SPI机制在JDK的DriverManagerSpringDubbo中得到了充分的利用,Dubbo中更是扩展了SPI机制来实现组件的可扩展性。

SPI在JDKDriverManager中的使用

mysql-connectorojdbc的jar包中,可以发现在META-INF/services目录下有一个名为java.sql.Driver的文件,在mysql-connectorjar包下,文件内容为:

com.mysql.cj.jdbc.Driver

这里就是定义了java.sql.Driver接口的实现类为com.mysql.cj.jdbc.Driver,在java.sql.DriverManager中,通过java.util.ServiceLoader来获取实现类,并实现调用。

        AccessController.doPrivileged(new PrivilegedAction<Void>() {
            public Void run() {

                ServiceLoader<Driver> loadedDrivers = ServiceLoader.load(Driver.class);
                Iterator<Driver> driversIterator = loadedDrivers.iterator();

                / Load these drivers, so that they can be instantiated.
                  It may be the case that the driver class may not be there
                  i.e. there may be a packaged driver with the service class
                  as implementation of java.sql.Driver but the actual class
                  may be missing. In that case a java.util.ServiceConfigurationError
                  will be thrown at runtime by the VM trying to locate
                  and load the service.
                 
                  Adding a try catch block to catch those runtime errors
                  if driver not available in classpath but it's
                  packaged as service and that service is there in classpath.
                 /
                try{
                    while(driversIterator.hasNext()) {
                        driversIterator.next();
                    }
                } catch(Throwable t) {
                // Do nothing
                }
                return null;
            }
        });

        println("DriverManager.initialize: jdbc.drivers = " + drivers);

        if (drivers == null || drivers.equals("")) {
            return;
        }
        String[] driversList = drivers.split(":");
        println("number of Drivers:" + driversList.length);
        for (String aDriver : driversList) {
            try {
                println("DriverManager.Initialize: loading " + aDriver);
                Class.forName(aDriver, true,
                        ClassLoader.getSystemClassLoader());
            } catch (Exception ex) {
                println("DriverManager.Initialize: load failed: " + ex);
            }
        }

 

2、Dubbo中的SPI扩展

Dubbo中扩展了ServiceLoaderExtentionLoader,来加载接口的实现类并维护其生命周期。

定义@SPI注解来标识扩展点的名称,表示可以该接口可以被ExtentionLoader类来加载,接口中的value值表示默认实现。

定义@Adaptive注解表示方法是一个自适应方法。在调用时会根据方法的参数来决定调用哪个具体的实现类。

Dubbo也扩展了Java SPI的目录。Dubbo会从以下目录中读取扩展配置信息:

  • META-INF/dubbo/internal
  • META-INF/dubbo
  • META-INF/services

如LoadBalance接口:

@SPI(RandomLoadBalance.NAME)
public interface LoadBalance {

    @Adaptive("loadbalance")
    <T> Invoker<T> select(List<Invoker<T>> invokers, URL url, Invocation invocation) throws RpcException;
}

这里RandomLoadBalance.NAME的值为random,在META-INF/dubbo/internal/com.alibaba.dubbo.rpc.cluster.LoadBalance文件中配置了该接口的实现类:

random=com.alibaba.dubbo.rpc.cluster.loadbalance.RandomLoadBalance

在调用时通过ExtentionLoader来获取实现类:

LoadBalance lb = ExtensionLoader.getExtensionLoader(LoadBalance.class).getExtension(loadbalanceName);

3、Spring中接口多实现调用

使用@Qualifier注解

Spring中@Service提供了value属性,来区分服务名称。并可以通过@Qualifier指定注入的服务;

如定义如下接口:

public interface PayService {

    void pay();
}

分别有如下实现:

@Service("aliPayService")
public class AliPayService implements PayService{

    @Override
    public void pay() {
        // ...
    }
}
@Service("wxPayService")
public class WxPayService implements PayService{

    @Override
    public void pay() {
        // ...
    }
}

在调用的时候就可以使用@ Qualifier指定注入的服务:

@Autowired
@Qualifier("wxPayService")
private PayService payService;

4、使用工厂模式

通过ApplicationContextgetBeansOfType获取接口所有实现类并放入容器中,在调用时动态获取实现类;

如定义如下接口:

public interface RemoteLockerService {

    /
      获取锁设备厂商
     
      @return 锁设备厂商
     /
    LockerManufacturerEnum getLockerManufacturer();

    /
      解锁
     
      @param identify 锁唯一标识
     /
    void unLock(String identify);
}

 

注入容器:

@Component
public class RemoteLockerServiceFactory implements ApplicationContextAware {

    private static Map<LockerManufacturerEnum, RemoteLockerService> lockerServiceMap;


    @Override
    public void setApplicationContext(ApplicationContext applicationContext) {
        lockerServiceMap = new HashMap<>();
        Map<String, RemoteLockerService> map = applicationContext
            .getBeansOfType(RemoteLockerService.class);
        map.forEach((key, value) -> lockerServiceMap.put(value.getLockerManufacturer(), value));
    }

    public static <T extends RemoteLockerService> T getRemoteLockerService(
        LockerManufacturerEnum lockerManufacturer) {
        return (T) lockerServiceMap.get(lockerManufacturer);
    }
}

调用时:

RemoteLockerService remoteLockerService = RemoteLockerServiceFactory.getRemoteLockerService(locker.getManufacturer());
remoteLockerService.unLock(identify);

 

  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Java SPI(Service Provider Interface)是一种针对接口编程的机制,允许第三方为接口提供实现实现类可以动态地被替换,从而实现热插拔的效果。 责任链模式是一种行为型设计模式,它将请求的发送者和接收者解耦,使多个对象都有机会处理该请求。在责任链模式,每个处理类都有一个后继处理类,如果当前处理类无法处理该请求,就将其转发给后继处理类。 利用Java SPI机制实现责任链模式的处理类热插拔的过程如下: 1. 定义一个接口,作为责任链每个处理类的基类。 ```java public interface Handler { void handle(Request request, Response response, HandlerChain chain); } ``` 2. 实现接口的具体处理类,每个处理类都有一个后继处理类。 ```java public class AuthenticationHandler implements Handler { @Override public void handle(Request request, Response response, HandlerChain chain) { // 处理请求 if (request.isAuthenticated()) { // 如果已认证,就将请求转发给下一个处理类 chain.next(request, response); } else { // 否则,直接返回未认证错误 response.setError("401 Unauthorized"); } } } public class AuthorizationHandler implements Handler { @Override public void handle(Request request, Response response, HandlerChain chain) { // 处理请求 if (request.isAuthorized()) { // 如果已授权,就将请求转发给下一个处理类 chain.next(request, response); } else { // 否则,直接返回未授权错误 response.setError("403 Forbidden"); } } } public class RateLimitHandler implements Handler { @Override public void handle(Request request, Response response, HandlerChain chain) { // 处理请求 if (request.isWithinRateLimit()) { // 如果未超过限制,就将请求转发给下一个处理类 chain.next(request, response); } else { // 否则,直接返回超过限制错误 response.setError("429 Too Many Requests"); } } } ``` 3. 定义一个HandlerChain类,用来维护责任链的处理类,以及处理请求的方法。 ```java public class HandlerChain { private List<Handler> handlers; private int index; public HandlerChain() { handlers = new ArrayList<>(); index = 0; } public void addHandler(Handler handler) { handlers.add(handler); } public void next(Request request, Response response) { // 如果还有后继处理类,就将请求转发给它 if (index < handlers.size()) { Handler handler = handlers.get(index++); handler.handle(request, response, this); } } public void handle(Request request, Response response) { // 将请求转发给第一个处理类 if (handlers.size() > 0) { Handler handler = handlers.get(0); handler.handle(request, response, this); } } } ``` 4. 在META-INF/services目录下创建一个文件,文件名为接口的全限定名,文件内容为实现类的全限定名,多个实现类用换行符分隔。 ``` com.example.Handler com.example.AuthenticationHandler com.example.AuthorizationHandler com.example.RateLimitHandler ``` 5. 在客户端代码,通过ServiceLoader加载所有实现类,并添加到HandlerChain。 ```java HandlerChain chain = new HandlerChain(); ServiceLoader<Handler> loader = ServiceLoader.load(Handler.class); for (Handler handler : loader) { chain.addHandler(handler); } Request request = new Request(); Response response = new Response(); chain.handle(request, response); ``` 这样,就可以实现对责任链模式的处理类进行热插拔了。如果需要添加或替换某个处理类,只需要将新的实现类打包成jar包,并将jar包放到classpath,就可以实现动态热插拔。 ### 回答2: Java SPI(Service Provider Interface)是一种标准的服务发现机制。通过SPI机制,可以动态地加载和替换代码的某些模块,实现了热插拔的功能。 责任链模式是一种行为型设计模式,它允许通过一系列的处理对象来逐步处理请求,并且每个处理对象都有机会处理请求或将其传递给下一个处理对象。SPI机制可以很好地与责任链模式结合,实现处理类的热插拔。 在Java SPI,首先需要定义一个接口,该接口定义了一系列的处理方法。然后,在代码使用SPI机制加载实现了该接口的具体处理类。通过SPI机制,可以在运行时动态地加载不同的处理类,实现责任链模式处理类的热插拔。 具体实现步骤如下: 1. 定义一个接口,例如"Handler",该接口包含一系列处理方法。 2. 创建不同的实现类,例如"HandlerA"、"HandlerB"、"HandlerC"等,这些实现类分别实现了"Handler"接口。 3. 在项目的resources目录下创建一个"META-INF/services"文件夹。 4. 在"META-INF/services"文件夹下创建一个以"Handler"接口全限定名命名的文件,例如"com.example.Handler"。 5. 在该文件,将实现了"Handler"接口的具体类的全限定名逐行添加进去,例如"com.example.HandlerA"、"com.example.HandlerB"、"com.example.HandlerC"。 6. 通过SPI机制,可以通过以下代码获取到实现了"Handler"接口的具体类实例: ```java ServiceLoader<Handler> handlers = ServiceLoader.load(Handler.class); ``` 7. 遍历handlers即可得到实现了"Handler"接口的具体类的实例,可以根据需要调用不同的处理方法。 通过SPI机制实现责任链模式的处理类热插拔,可以使系统更加灵活和可扩展。通过配置不同的实现类,可以实现动态地修改和扩展处理类的功能,而无需修改和重新编译源代码。同时,SPI机制还符合开闭原则,提高了代码的可维护性和可扩展性。 ### 回答3: Java SPI(Service Provider Interface)是Java提供的一种服务提供者接口,它可以实现在运行时动态地加载和替换实现类的功能。而责任链模式是一种设计模式,它通过将一个请求经过一系列处理对象的处理,直到找到合适的处理者为止。 利用Java SPI机制实现责任链模式的处理类热插拔,可以通过以下步骤完成: 1. 定义接口:首先需要定义一个处理请求的接口,该接口包含一个处理方法,用于处理具体的请求。 2. 实现接口:根据需求,实现多个处理类,每个类都实现上述定义的接口,并编写相应的处理逻辑。 3. 创建配置文件:在资源目录下创建一个META-INF/services文件夹,并在其创建一个以接口全限定名为名称的文件,文件内容为实现类的全限定名,每个实现类占据一行。 4. 加载实现类:在代码通过调用ServiceLoader.load()方法,加载指定接口的所有实现类。这样,就可以动态地获取到所有实现类的实例。 5. 构建责任链:根据加载到的实现类实例,按照需要的顺序构建责任链。责任链的每个节点都是一个实现类的实例。 6. 处理请求:将请求传递给责任链的第一个节点,由节点依次处理请求,直到找到合适的处理者或责任链结束。 通过上述步骤,就可以实现在运行时动态地添加、删除或替换处理类,从而实现责任链模式处理类的热插拔。 利用Java SPI机制实现责任链模式的处理类热插拔的好处是可以在不修改现有代码的情况下,通过添加或删除实现类来实现不同的业务逻辑处理。这种解耦的设计模式可以提高代码的可维护性和扩展性。同时,由于Java SPI机制利用了类加载器来加载实现类,可以方便地实现实现类的动态加载和替换,使得代码更加灵活和可配置。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值