Dubbo为什么不直接使用Java的SPI机制?

简介: 而要解决上面这个问题,就涉及到了Dubbo的SPI机制,它是Dubbo的扩展点,通过它我们能清楚的知道Dubbo底层真正使用的第三方框架及原理实现。

01提出问题


其实会提出这个问题,也是因为强哥在知乎上被问了另一个问题:


0.png


而要解决上面这个问题,就涉及到了Dubbo的SPI机制,它是Dubbo的扩展点,通过它我们能清楚的知道Dubbo底层真正使用的第三方框架及原理实现。


但是,我们又知道,其实Java也有自己的一套SPI,作为JDK内置的一种服务提供发现机制。而目前市面上很多框架都用它来做服务的扩展发现。SPI的简单理解就是一种动态替换发现的机制。举个例子,我们想在运行时动态给它添加实现,你只需要添加一个实现,然后把新的实现描述给JDK知道就行了。大家耳熟能详的如JDBC,日志框架都有用到。

那么为什么Dubbo不用Java的SPI而要自己搞一套呢?那还用说,当然是因为其本身存在一定的弊端,导致Dubbo不得不自己再造个新轮子。


Java SPI的使用


我们先来看看怎么使用Java的SPI。


  1. 需要在 classpath 下创建一个目录,该目录命名必须是:META-INF/service;
  2. 在该目录下创建一个 properties 文件,该文件需要满足以下几个条件
    2.1 文件名必须是扩展的接口的全路径名称
    2.2 文件内部描述的是该扩展接口的所有实现类
    2.3 文件的编码格式是 UTF-8
  3. 通过 java.util.ServiceLoader 的加载机制来发现


不多啰嗦,直接给例子,文件目录如下:


1.png


代码如下,为了简单,把各个文件代码合到一起展示:


public interface INanShen {    void show();}
public class DianZanQiangGe implements INanShen{    public void show() {        System.out.println("欢迎点赞,强哥叨逼叨!");    }}
public class GuanZhuQiangGe implements INanShen{    public void show() {        System.out.println("欢迎关注,强哥叨逼叨!");    }}
public class NiuBiMain {
    public static void main(String[] args) {        ServiceLoader<INanShen> serviceLoader = ServiceLoader.load(INanShen.class);        Iterator<INanShen> iterator = serviceLoader.iterator();        while (iterator.hasNext()) {            INanShen nanShen = iterator.next();            nanShen.show();        }    }}


文件demo.INanShen内容及运行对应输出如下:


内容:demo.GuanZhuQiangGe输出:欢迎关注,强哥叨逼叨!
内容:demo.GuanZhuQiangGedemo.DianZanQiangGe输出:欢迎关注,强哥叨逼叨!欢迎点赞,强哥叨逼叨!


看起来是不是非常简单好用,那么为什么Dubbo不直接使用呢?

这就要从源码角度找原因了,感兴趣的同学可以复制上面的代码debug走一下Java SPI源码的加载流程,限于篇幅,强哥就不列出源码了,流程大体如下:


2.png


简单来说就是:就是约定一个目录META-INF/services/,根据接口名去那个目录找到文件,文件解析得到实现类的全限定名,然后循环加载实现类和创建其实例。这种方式存在一些缺点:


  1. JDK 标准的 SPI 会一次性加载实例化扩展点的所有实现,什么意思呢?就是如果你在 META-INF/service 下的文件里面加了 N 个实现类,那么 JDK 启动的时候都会一次性全部加载。那么如果有的扩展点实现初始化很耗时或者如果有些实现类并没有用到, 那么会很浪费资源;
  2. 如果扩展点加载失败,会导致调用方报错,而且这个错误很难定位到是这个原因。


所以,Dubbo为了避免以上的确定,自己搞了一套更牛的SPI机制。


02问题解析


Dubbo的SPI机制,主要用了它自己的一个加载器ExtensionLoader。除了可以按需加载实现类之外,增加了 IOC 和 AOP 的特性,还有个自适应扩展机制。


对配置文件目录的约定,不同于 Java SPI ,Dubbo 分为了三类目录:


  1. META-INF/services/ 目录:该目录下的 SPI 配置文件是为了用来兼容 Java SPI 。
  2. META-INF/dubbo/ 目录:该目录存放用户自定义的 SPI 配置文件。
  3. META-INF/dubbo/internal/ 目录:该目录存放 Dubbo 内部使用的 SPI 配置文件。


而其配置文件的内容,也改成了键值对的方式。


可能这么说大家一下子还不太好理解,我们来通过解决上面知乎网友问的问题,来加深一下,问题是:Dubbo真的底层是基于Netty吗?


怎么知道呢?从上面的三个目录中我们找一下,可以在META-INF/dubbo/internal/下找到如下文件:


3.png


内容为:


4.png


然后全局搜索文件名对应的类:org.apache.dubbo.remoting.Transporter


5.png


嘿嘿,从Transporter类上的注解@SPI("netty"),我们便可以知道,Dubbo底层传输默认使用的是netty,不是netty3,netty4也不是mina。


而具体的使用,有兴趣的小伙伴也可以看看Transporter类下的bind和connect的子类实现,也就是netty的实现:


6.png


终上所述,也就解答了:Dubbo底层确实是基于Netty的。


相关文章
|
设计模式 人工智能 安全
AQS:Java 中悲观锁的底层实现机制
AQS(AbstractQueuedSynchronizer)是Java并发包中实现同步组件的基础工具,支持锁(如ReentrantLock、ReadWriteLock)和线程同步工具类(如CountDownLatch、Semaphore)等。Doug Lea设计AQS旨在抽象基础同步操作,简化同步组件构建。 使用AQS需实现`tryAcquire(int arg)`和`tryRelease(int arg)`方法以获取和释放资源,共享模式还需实现`tryAcquireShared(int arg)`和`tryReleaseShared(int arg)`。
538 32
AQS:Java 中悲观锁的底层实现机制
|
12月前
|
人工智能 Java 关系型数据库
Java——SPI机制详解
SPI(Service Provider Interface)是JDK内置的服务提供发现机制,主要用于框架扩展和组件替换。通过在`META-INF/services/`目录下定义接口实现类文件,Java程序可利用`ServiceLoader`动态加载服务实现。SPI核心思想是解耦,允许不同厂商为同一接口提供多种实现,如`java.sql.Driver`的MySQL与PostgreSQL实现。然而,SPI存在缺陷:需遍历所有实现并实例化,可能造成资源浪费;获取实现类方式不够灵活;多线程使用时存在安全问题。尽管如此,SPI仍是Java生态系统中实现插件化和模块化设计的重要工具。
676 0
|
10月前
|
人工智能 前端开发 安全
Java开发不可不知的秘密:类加载器实现机制
类加载器是Java中负责动态加载类到JVM的组件,理解其工作原理对开发复杂应用至关重要。本文详解类加载过程、双亲委派模型及常见类加载器,并介绍自定义类加载器的实现与应用场景。
372 4
|
Java 区块链 网络架构
酷阿鲸森林农场:Java 区块链系统中的 P2P 区块同步与节点自动加入机制
本文介绍了基于 Java 的去中心化区块链电商系统设计与实现,重点探讨了 P2P 网络在酷阿鲸森林农场项目中的应用。通过节点自动发现、区块广播同步及链校验功能,系统实现了无需中心服务器的点对点网络架构。文章详细解析了核心代码逻辑,包括 P2P 服务端监听、客户端广播新区块及节点列表自动获取等环节,并提出了消息签名验证、WebSocket 替代 Socket 等优化方向。该系统不仅适用于农业电商,还可扩展至教育、物流等领域,构建可信数据链条。
|
缓存 Dubbo Java
理解的Java中SPI机制
本文深入解析了JDK提供的Java SPI(Service Provider Interface)机制,这是一种基于接口编程、策略模式与配置文件组合实现的动态加载机制,核心在于解耦。文章通过具体示例介绍了SPI的使用方法,包括定义接口、创建配置文件及加载实现类的过程,并分析了其原理与优缺点。SPI适用于框架扩展或替换场景,如JDBC驱动加载、SLF4J日志实现等,但存在加载效率低和线程安全问题。
694 7
理解的Java中SPI机制
|
12月前
|
人工智能 JavaScript Java
Java反射机制及原理
本文介绍了Java反射机制的基本概念、使用方法及其原理。反射在实际项目中比代理更常用,掌握它可以提升编程能力并理解框架设计原理。文章详细讲解了获取Class对象的四种方式:对象.getClass()、类.class、Class.forName()和类加载器.loadClass(),并分析了Class.forName()与ClassLoader的区别。此外,还探讨了通过Class对象进行实例化、获取方法和字段等操作的具体实现。最后从JVM类加载机制角度解析了Class对象的本质及其与类和实例的关系,帮助读者深入理解Java反射的工作原理。
299 0
|
存储 Java 编译器
Java 中 .length 的使用方法:深入理解 Java 数据结构中的长度获取机制
本文深入解析了 Java 中 `.length` 的使用方法及其在不同数据结构中的应用。对于数组,通过 `.length` 属性获取元素数量;字符串则使用 `.length()` 方法计算字符数;集合类如 `ArrayList` 采用 `.size()` 方法统计元素个数。此外,基本数据类型和包装类不支持长度属性。掌握这些区别,有助于开发者避免常见错误,提升代码质量。
1093 1
|
缓存 运维 Java
Java静态代码块深度剖析:机制、特性与最佳实践
在Java中,静态代码块(或称静态初始化块)是指类中定义的一个或多个`static { ... }`结构。其主要功能在于初始化类级别的数据,例如静态变量的初始化或执行仅需运行一次的初始化逻辑。
483 4
|
Dubbo Java 应用服务中间件
微服务学习 | Springboot整合Dubbo+Nacos实现RPC调用
微服务学习 | Springboot整合Dubbo+Nacos实现RPC调用
|
Dubbo Java 应用服务中间件
Spring Cloud Dubbo:微服务通信的高效解决方案
【10月更文挑战第15天】随着信息技术的发展,微服务架构成为企业应用开发的主流。Spring Cloud Dubbo结合了Dubbo的高性能RPC和Spring Cloud的生态系统,提供高效、稳定的微服务通信解决方案。它支持多种通信协议,具备服务注册与发现、负载均衡及容错机制,简化了服务调用的复杂性,使开发者能更专注于业务逻辑的实现。
408 2