Dubbo是如何基于动态代理实现RPC调用的?

以前用 SpringCloud 的时候,你要调用别的系统一般用的是 Feign 这个框架,然后现在你
首页 新闻资讯 行业资讯 Dubbo是如何基于动态代理实现RPC调用的?

今天给大家讲一个知识点,就是平时我们很多兄弟开发的系统都不再是那种 10 年前的简单单块系统了,一个工程打包部署启动,系统连接 MySQL,然后 Crud 整起就够的了,我们现在开发的系统都是很高大上的分布式系统。

啥叫分布式系统?

就是说你写的系统收到一个请求之后,你自己的代码跑完还不够,你得去调用别的兄弟写的系统,让他的系统也干一些事儿,然后他的活儿也干完了之后,你这次请求处理才算是完事儿了。

就因为你处理请求得调用别的兄弟系统一起运行,一个请求涉及到了分布在多台机器上的多个系统,所以就叫做分布式了。

如下图:

77d40bb76ecbdd53cfe104ffe23f4edc82b917.png

分布式系统之间如何调用呢?

那现在兄弟们平时开发分布式系统,就是去调用别的系统,一般都是用什么框架呢?

简单,现在兄弟们一般都是用 SpringCloud,或者是用 Dubbo,这两种都有人用,用 SpringCloud 的一般前两年多一些,最近这两年大家都纷纷转用 SpringCloud Alibaba 了。

以前用 SpringCloud 的时候,你要调用别的系统一般用的是 Feign 这个框架,然后现在你用 SpringCloud Alibaba 的时候,一般用的都是 Dubbo 这个框架,我们今天就以 Dubbo 这个框架举例来讲讲我们平时系统之间是如何进行调用的。

首先呢,我们还是看上面那个图里的业务系统 B,这个系统如果要提供接口给别人调用,那么他必须写一个接口,这个接口里得定义好你要允许别人调用哪些方法,大致看起来可能类似下面这样的代码。

如下:

复制

public interface Service {  
    String sayHello(String name);   
}

接着呢,你得针对这个接口开发一个实现类,实现类里需要完成这个方法的逻辑,同时还得给这个实现类加上@DubboService这个注解,让Dubbo把他识别为一个对外的服务接口,如下面的代码:
@DubboService(version = "1.0.0", interfaceClass = Service.class)
public class ServiceImpl implements Service {    
    public String sayHello(String name) {   
        // 运行一些代码    
      return "hello, " + name;   }  

}
  • 1.

  • 2.

  • 3.

  • 4.

  • 5.

  • 6.

  • 7.

  • 8.

  • 9.

  • 10.

  • 11.

  • 12.

那么当你的业务系统 B 开发好上面的接口和实现类,同时加上了 @DubboService 这个注解之后,这个业务系统 B 启动以后,会干一个什么事儿呢?

简单来说,Dubbo 框架会随着你的业务系统 B 一起启动,他会启动一个网络服务器,这个网络服务器会监听一个你指定的端口号,通常这个端口号是 20880 端口。

如下图:

33b467d285085485bdf009a9bb7124ea2235f5.png

这个时候业务系统 B 上的 Dubbo 已经启动好了网络服务器监听了一个端口号,随时可以接收你发送过来的调用请求。

接下来就轮到咱们的业务系统 A 出场了,这个业务系统 A 假设要调用业务系统 B 的 Service 接口中定义的那些方法,他会怎么做呢?

这个代码大概会是这样的:

复制

@RestController 
public class Controller {  
    // 注意,这里的Service就是业务系统B定义的接口     
    @DubboReference(version = "1.0.0")  
    private Service service;    


    @RequestMapping("/hello")  
    public Response sayHello(String name) {          
        String result = service.sayHello(name);    
        return Response.success(result);   }    
}
  • 1.

  • 2.

  • 3.

  • 4.

  • 5.

  • 6.

  • 7.

  • 8.

  • 9.

  • 10.

  • 11.

  • 12.

所以说,这里最关键的问题来了,上面是业务系统 A 的代码,他仅仅是定义了一个业务系统 B 的 Service 接口的变量。

就是 Service service 这个变量,然后加了一个 @DubboReference 注解,所以这个业务系统 A 启动的时候,Dubbo 又会干点什么事儿呢?

Dubbo 是如何基于动态代理实现 RPC 调用的

其实这里有一个很重点的点,那就是 Dubbo 此时会使用我们设计模式里的代理模式,去创建一个动态代理对象,把这个动态代理对象注入给我们上面的 Service service 这个变量,让他那个变量引用 Dubbo 的动态代理对象。

那么这个动态代理对象是个什么东西呢?简单来说,就是 Dubbo 可以动态生成一个类,这个类是实现了 Service 接口的,然后所有的方法都是有他自己的一套实现逻辑的。

具体什么实现逻辑一会儿我们再说,但是现在看起来应该如下图:

f5c84d982bfee5c0ac32215ae7cd022714e487.png

所以这里其实很关键的一点是,大家一定要在这里理解这个 Dubbo 动态代理的概念,这是设计模式中代理模式一个很经典的运用。

就是说,一旦 Dubbo 生成了针对接口的动态代理对象,注入给了 Service service 这个变量,那么你业务系统 A 里调用 Service service 的方法时,其实是会调用 Dubbo 动态代理对象的方法的。

再看一下代码感受一下:

复制

@RestController 
public class Controller {     

    // 注意,这里的Service就是业务系统B定义的接口   
    // 这个接口变量其实会被注入Dubbo生成的动态代理对象     
    @DubboReference(version = "1.0.0")   
    private Service service;   

    @RequestMapping("/hello")   
    public Response sayHello(String name) { 
        // 注意,这里你调用接口方法的时候,其实是在调用Dubbo动态代理对象的方法   
        String result = service.sayHello(name);    
     return Response.success(result);
    }  

}
  • 1.

  • 2.

  • 3.

  • 4.

  • 5.

  • 6.

  • 7.

  • 8.

  • 9.

  • 10.

  • 11.

  • 12.

  • 13.

  • 14.

  • 15.

  • 16.

接着 Dubbo 动态代理对象的方法被调用的时候,他会干什么事情呢?

其实这里他就会跟我们的业务系统 B 所在的机器建立一个网络连接,然后通过这个网络连接把一个调用请求发送过去。

业务系统 B 里面的 Dubbo 网络服务器收到请求之后,就会根据请求调用本地的接口实现类的方法,拿到返回值,接着通过网络连接把返回值返回给业务系统 A 的 Dubbo 动态代理对象,最后,Dubbo 动态代理对象就会把这个返回值交给我们了。

如下图:

012dcce5776563e153250368a9a0339da136e6.png

好了,今天给大家分享的基于 Dubbo 实现系统间调用的原理就到这里了,希望大家平时用 Dubbo 做开发的时候,对他底层的原理也得有一定的理解。