微服务-java spi 与 dubbo spi

Java SPI

通过一个案例来看SPI

java 复制代码
public interface DemoSPI {
    void echo();
}
public class FirstImpl implements DemoSPI{
    @Override
    public void echo() {
        System.out.println("first echo");
    }
}
public class SecondImpl implements DemoSPI{
    @Override
    public void echo() {
        System.out.println("second echo");
    }
}
public class Main {
    public static void main(String[] args) {
        ServiceLoader<DemoSPI> loader = ServiceLoader.load(DemoSPI.class);
        Iterator<DemoSPI> iterator = loader.iterator();
        while (iterator.hasNext()) {
          iterator.next().echo();
        }
        System.out.println("加载完毕!!!!!");
    }
}

SPI :Service Provider Interface, 服务提供接口。

JDK SPI:ServiceLoader.load

使用load方法频率高,容易影响IO吞吐和内存消耗, 可以看到回去静态资源目录下读取文件。

使用load方法想要获取指定的实现类,需要自己进行遍历并编写各种比较代码,可能实现有多个所以需要比对。

Dubbo SPI: ApplicationModel.defaultModel().getExtensionLoader

增加缓存,来降低磁盘IO访问及减少对象的生成

使用Map的hash查找,来提升检索指定实现类的性能。

java 复制代码
@SPI
public interface DubboSPI {
    void echo();
}
public class DubboFirstSPI implements DubboSPI{
    @Override
    public void echo() {
        System.out.println("first dubbo spi");
    }
}
public class DubboSecondSPI implements DubboSPI{
    @Override
    public void echo() {
        System.out.println("second spi");
    }
}

public class DubboMain {
    public static void main(String[] args) {
        ExtensionLoader<DubboSPI> loader = ExtensionLoader.getExtensionLoader(DubboSPI.class);
        DubboSPI firstf = loader.getExtension("first");
        DubboSPI second = loader.getExtension("second");
        firstf.echo();
        second.echo();
    }
}

com.liyong.leran.dubbospi.DubboSPI里面的内容:

first=com.liyong.leran.dubbospi.DubboFirstSPI

second=com.liyong.leran.dubbospi.DubboSecondSPI

java 复制代码
public T getExtension(String name, boolean wrap) {
      if (StringUtils.isEmpty(name)) {
          throw new IllegalArgumentException("Extension name == null");
      }
      if ("true".equals(name)) {
          return getDefaultExtension();
      }
      final Holder<Object> holder = getOrCreateHolder(name);
      Object instance = holder.get();
      if (instance == null) {
          synchronized (holder) {
              instance = holder.get();
              if (instance == null) {
                  instance = createExtension(name, wrap);
                  holder.set(instance);
              }
          }
      }
      return (T) instance;
  }
java 复制代码
// 走缓存的方法
private Holder<Object> getOrCreateHolder(String name) {
      Holder<Object> holder = cachedInstances.get(name);
      if (holder == null) {
          cachedInstances.putIfAbsent(name, new Holder<>());
          holder = cachedInstances.get(name);
      }
      return holder;
  }
相关推荐
fantasy_423 分钟前
LeetCode238☞除自身以外数组的乘积
java·数据结构·python·算法·leetcode
GalenZhang88826 分钟前
Java生成微信小程序码及小程序短链接
java·微信小程序·小程序
元亓亓亓29 分钟前
Java后端开发day38--不可变集合&Stream流
java·开发语言
纪元A梦36 分钟前
华为OD机试真题——阿里巴巴找黄金宝箱Ⅰ(2025A卷:100分)Java/python/JavaScript/C/C++/GO最佳实现
java·c语言·javascript·c++·python·华为od·go
CHQIUU1 小时前
跨语言哈希一致性:C# 与 Java 的 MD5 之战?
java·c#·哈希算法
JavaDog程序狗1 小时前
【java】easypoi导出excel单元格,填充动态下拉列
java·spring boot·excel
Java中文社群1 小时前
国内首个「混合推理模型」Qwen3深夜开源,盘点它的N种对接方式!
java·人工智能·后端
掉鱼的猫1 小时前
qwen3 惊喜发布,用 ollama + solon ai (java) 尝个鲜
java·openai·deepseek
GuGuStudy1 小时前
这都是什么多线程知识
java
Bob99981 小时前
Amlogic S905L3系列盒子 ROM DIY相关
java·javascript·数据仓库·vscode·eclipse·tomcat·vim