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


相关文章
|
18天前
|
Java 程序员
深入理解Java异常处理机制
Java的异常处理是编程中的一块基石,它不仅保障了代码的健壮性,还提升了程序的可读性和可维护性。本文将深入浅出地探讨Java异常处理的核心概念、分类、处理策略以及最佳实践,旨在帮助读者建立正确的异常处理观念,提升编程效率和质量。
|
19天前
|
Java 开发者 UED
深入探索Java中的异常处理机制##
本文将带你深入了解Java语言中的异常处理机制,包括异常的分类、异常的捕获与处理、自定义异常的创建以及最佳实践。通过具体实例和代码演示,帮助你更好地理解和运用Java中的异常处理,提高程序的健壮性和可维护性。 ##
42 2
|
19天前
|
Java 开发者
Java中的异常处理机制深度剖析####
本文深入探讨了Java语言中异常处理的重要性、核心机制及其在实际编程中的应用策略,旨在帮助开发者更有效地编写健壮的代码。通过实例分析,揭示了try-catch-finally结构的最佳实践,以及如何利用自定义异常提升程序的可读性和维护性。此外,还简要介绍了Java 7引入的多异常捕获特性,为读者提供了一个全面而实用的异常处理指南。 ####
39 2
|
22天前
|
Java 程序员 UED
深入理解Java中的异常处理机制
本文旨在揭示Java异常处理的奥秘,从基础概念到高级应用,逐步引导读者掌握如何优雅地管理程序中的错误。我们将探讨异常类型、捕获流程,以及如何在代码中有效利用try-catch语句。通过实例分析,我们将展示异常处理在提升代码质量方面的关键作用。
31 3
|
22天前
|
Java 数据库连接 开发者
Java中的异常处理机制:深入解析与最佳实践####
本文旨在为Java开发者提供一份关于异常处理机制的全面指南,从基础概念到高级技巧,涵盖try-catch结构、自定义异常、异常链分析以及最佳实践策略。不同于传统的摘要概述,本文将以一个实际项目案例为线索,逐步揭示如何高效地管理运行时错误,提升代码的健壮性和可维护性。通过对比常见误区与优化方案,读者将获得编写更加健壮Java应用程序的实用知识。 --- ####
|
23天前
|
运维 Java 编译器
Java 异常处理:机制、策略与最佳实践
Java异常处理是确保程序稳定运行的关键。本文介绍Java异常处理的机制,包括异常类层次结构、try-catch-finally语句的使用,并探讨常见策略及最佳实践,帮助开发者有效管理错误和异常情况。
69 4
|
22天前
|
开发框架 安全 Java
Java 反射机制:动态编程的强大利器
Java反射机制允许程序在运行时检查类、接口、字段和方法的信息,并能操作对象。它提供了一种动态编程的方式,使得代码更加灵活,能够适应未知的或变化的需求,是开发框架和库的重要工具。
36 2
|
17天前
|
Java API 开发者
深入理解Java中的异常处理机制
本文探讨了Java编程语言中异常处理的核心概念,包括异常类型、异常捕获与抛出、以及最佳实践。通过分析常见的异常场景和处理策略,旨在帮助开发者更好地理解和运用异常处理机制,提高代码的健壮性和可维护性。文章不仅涵盖了基本的try-catch结构,还深入讨论了自定义异常的创建与使用,以及finally块的重要性和应用。此外,还将介绍一些高级技巧,如多异常捕获和嵌套异常处理,为读者提供全面的技术指导。
68 0
|
24天前
|
Java API 数据库
Java 反射机制:动态编程的 “魔法钥匙”
Java反射机制是允许程序在运行时访问类、方法和字段信息的强大工具,被誉为动态编程的“魔法钥匙”。通过反射,开发者可以创建更加灵活、可扩展的应用程序。
39 0
|
24天前
|
Java 开发者
Java中的异常处理机制####
本文深入探讨了Java编程语言中异常处理的核心概念,旨在为开发者提供一套清晰的异常管理策略。不同于传统的摘要概述,本文将通过一个实际案例,逐步揭示如何有效地捕获、处理和抛出异常,以及如何利用自定义异常来增强程序的健壮性和可读性。我们将从基础的try-catch语句出发,逐步深入到finally块的使用,再到throws和throw关键字的区别,最后展示如何创建自定义异常类,以应对特定错误情况。 ####
21 0