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的。


相关文章
|
6天前
|
Java
深入理解Java中的异常处理机制
【9月更文挑战第34天】在Java的世界里,异常处理是代码健壮性的守护神。本文将带你探索Java异常处理的奥秘,从基础语法到高级技巧,我们一步步揭开异常处理的面纱。你将学会如何捕获、声明和处理异常,以及如何自定义异常类型。让我们开始这段旅程,让代码更加稳健和可靠吧!
|
3天前
|
Java
深入浅出Java异常处理机制
【9月更文挑战第37天】在Java编程的世界里,异常处理是一项基础而重要的技能。它就像是我们生活中的急救箱,遇到意外时能及时救治,避免程序崩溃。本文将通过生动的例子和易于理解的语言,带你了解Java中的异常处理机制,从基本的try-catch语句到自定义异常的创建,让你的程序更加健壮和可靠。准备好了吗?让我们一起走进Java异常处理的大门,探索它的奥秘吧!
|
9天前
|
Java 程序员 开发者
深入理解Java中的异常处理机制
【9月更文挑战第31天】在Java编程中,异常处理是维护程序健壮性的关键。本文将通过浅显易懂的语言和生动的例子,带你了解Java异常处理的基本概念、分类以及如何优雅地处理它们。从初学者到资深开发者,每个人都能从中获得新的洞见和技巧,让你的代码更加健壮和易于维护。
12 4
|
8天前
|
Java 编译器 开发者
Java中的异常处理机制:从基础到进阶
本文深入探讨Java编程语言中的异常处理机制,从基础知识出发,逐步解析异常的分类、捕获和处理方法。通过实际案例分析,展示如何在开发过程中有效利用异常处理提高代码的稳定性和可维护性。进一步探讨了自定义异常的创建和使用场景,以及在Java中进行异常处理的最佳实践。文章旨在为Java开发者提供一个全面而详细的异常处理指南,帮助开发者更好地理解和运用Java的异常处理机制。
|
12天前
|
Java 数据库连接
深入理解Java异常处理机制
【9月更文挑战第28天】在Java编程中,异常处理是确保程序健壮性的关键。本文通过浅显易懂的语言和生动的例子,带你一步步了解Java的异常处理机制。从try-catch-finally的基本用法,到自定义异常类,再到异常处理的最佳实践,我们将一起探索如何在代码中优雅地处理那些不期而遇的小插曲。
16 4
|
14天前
|
Java 程序员 数据库连接
Java中的异常处理机制:理解与实践
本文将深入探讨Java语言中异常处理的核心概念、重要性以及应用方法。通过详细解析Java异常体系结构,结合具体代码示例,本文旨在帮助读者更好地理解如何有效利用异常处理机制来提升程序的健壮性和可维护性。
|
14天前
|
Java 开发者 UED
深入理解Java中的异常处理机制
本文旨在通过通俗易懂的语言,详细解析Java异常处理的核心概念及应用。从异常的基本分类到具体处理方法,再到最佳实践和常见误区,一步步引领读者深入理解这一关键技术,提升编程质量和效率。
18 2
|
14天前
|
Java 程序员 数据库连接
深入理解Java中的异常处理机制
【9月更文挑战第25天】在Java的海洋中航行,不可避免地会遇到异常的风暴。本文将作为你的航海图,指引你穿越异常处理的迷雾,让你学会如何使用try-catch语句、finally块以及throw和throws关键字来驾驭这些风暴。我们将一起探索自定义异常的岛屿,并了解如何创建和使用它们。准备好了吗?让我们启航,确保你的代码不仅能够抵御异常的狂澜,还能优雅地处理它们。
|
14天前
|
Java 开发者
Java中的异常处理机制深度解析
在Java编程中,异常处理是保证程序稳定性和健壮性的重要手段。本文将深入探讨Java的异常处理机制,包括异常的分类、捕获与处理、自定义异常以及一些最佳实践。通过详细讲解和代码示例,帮助读者更好地理解和应用这一机制,提升代码质量。
14 1
|
14天前
|
Java 开发者 UED
Java中的异常处理机制:理解与应用
本文深入探讨Java的异常处理机制,通过实例解析如何有效使用try-catch-finally块、throws关键字及自定义异常,以提升代码的健壮性和可维护性。我们将从基础概念入手,逐步过渡到高级应用,为Java开发者提供全面指导。