`
IXHONG
  • 浏览: 452989 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

dubbo请求调用过程分析

阅读更多

服务消费方发起请求

 

当服务的消费方引用了某远程服务,服务的应用方在spring的配置实例如下:

 

<dubbo:reference id="demoService" interface="com.alibaba.dubbo.demo.DemoService" />

 

demoService实例其实是代理工厂生产的代理对象(大家可以参考代理那部分生成的伪代码),在代码中调用demoService.sayHello(“world!”)时,

 

1.      将方法名方法参数传入InvokerInvocationHandler的invoke方

 

对于Object中的方法toString, hashCode, equals直接调用invoker的对应方法,

 

这里对于Object的方法需要被远程调用吗?调用了是不是报错比默认处理更好呢??

 

远程调用层是以Invocation, Result为中心, 这里根据要调用的方法以及传入的参数构建RpcInvocation对象,作为Invoker的入参

 

2.      MockClusterInvoker根据参数提供了三种调用策略

 

不需要mock, 直接调用FailoverClusterInvoker

 

强制mock,调用mock

 

先调FailoverClusterInvoker,调用失败在mock、

 

3.      FailoverClusterInvoker默认调用策略

 

通过目录服务查找到所有订阅的服务提供者的Invoker对象

 

路由服务根据策略来过滤选择调用的Invokers

 

通过负载均衡策略LoadBalance来选择一个Invoker

 

4.      执行选择的Invoker.inoker(invocation)

 

经过监听器链,默认没有

 

经过过滤器链,内置实现了很多

 

执行到远程调用的DubboInvoker

 

5.      DubboInvoker

 

根据url 也就是根据服务提供者的长连接,这里封装成交互层对象ExchangeClient供这里调用

 

判断远程调用类型同步,异步还是oneway模式

 

ExchangeClient发起远程调用,底层remoting不在这里描述了

 

获取调用结果:

 

        Oneway返回空RpcResult

 

        异步,直接返回空RpcResult, ResponseFuture回调

 

        同步, ResponseFuture模式同步转异步,等待响应返回

 

 

 

 

 

服务提供方接收调用请求

 

同样我们也是rpc调用层DubboProtocol层开始分析,对于通信层remoting的数据接收反序列等等过程不做分析。

 

DubboProtocol的requestHandler是ExchangeHandler的实现,是remoting层接收数据后的回调。

 

requestHandler.replay方法接收请求消息,这里只处理远程调用消息Invocation。

 

1.      通过Invocation获取服务名和端口组成serviceKey=com.alibaba.dubbo.demo.DemoService:20880, 从DubboProtocol的exproterMap中获取暴露服务的DubboExporter, 在从dubboExporter 获取invoker返回

 

2.      经过过滤器链

 

3.      经过监听器链

 

4.      到达执行真正调用的invoker, 这个invoker由代理工厂ProxyFactory.getInvoker(demoService, DemoService.class, registryUrl)创建,具体请看代理那部分介绍。

 

调用demoService实例方法,将结果封装成RpcResult返回

 

5.      交换层构建Response,通过Remoting层编码传输将结果响应给调用方

 

 

 

 

 

服务消费方发起远程调用的底层通信

 

 

服务提供方接收请求并响应的底层通信

 

一:provider提供方

 



 

ClassPathXmlApplicationContext <init>(构造方法)

 

-> ClassPathXmlApplicationContext refresh()

 

-> ClassPathXmlApplicationContext finishRefresh()

 

-> AbstractApplicationContext publishEvent()

 

-> ServiceBean onApplicationEvent()

 

-> ServiceConfig doExport()

 

#构造dubbo对象 application provider module protocol registry service reference consume等

 

 

 

-> ServiceConfig doExportUrls #导出URL,获取注册中心RegistryConfig

 

#注册中心:registry://10.199.101.228:2181/com.alibaba.dubbo.registry.RegistryService?application=demo&backup=10.199.101.227:2181,10.199.101.229:2181&dubbo=2.4.9&pid=8045&registry=zookeeper&timestamp=1491546077803

 

 

 

-> ServiceConfig doExportUrlsFor1Protocol()

 

#需要暴露 dubbo://10.199.66.242:20880/com.unj.dubbotest.provider.DemoService?anyhost=true&application=dubbo_demo_provider&dubbo=2.4.9&interface=com.unj.dubbotest.provider.DemoService&methods=sayHello,getUsers&pid=8045&revision=0.0.1&side=provider&timestamp=1491546674441&version=0.0.1

 

 

 

-> ServiceConfig exportLocal()

 

-> Exporter<?> exporter = protocol.export(proxyFactory.getInvoker(ref, (Class) interfaceClass, local));

 

#暴露Invoker<XxxService>调用服务代理类

 

 

 

-> proxyFactory.getInvoker(ref, (Class) interfaceClass, local)

 

#返回 AbstractProxyInvoker代理ProxyInvoker<XxxService>

 

public abstract class AbstractProxyInvoker<T> implements Invoker<T> {

 

private final T proxy; //代理目标实例 XxxServiceImpl

 

private final Class<T> type;

 

private final URL url;

 

}

 

-> InvokerInvocationHandler.invoke()

 

#invoker.invoke(new RpcInvocation(method, args)).recreate();

 

 

 

-> DubboProtocol export(Invoker<T> invoker)

 

# 返回暴露Exporter<T>

 

public class DubboExporter<T> extends AbstractExporter<T> {

 

private final String key; //com.unj.dubbotest.provider.DemoService:0.0.1:20880

 

private final Map<String, Exporter<?>> exporterMap;

 

public DubboExporter(Invoker<T> invoker, String key, Map<String, Exporter<?>> exporterMap){

 

super(invoker);

 

this.key = key;

 

this.exporterMap = exporterMap;

 

}

 

 

 

-> DubboProtocol openServer(url)

 

#url dubbo://10.199.66.242:20880/com.unj.dubbotest.provider.DemoService?anyhost=true&application=dubbo_demo&dubbo=2.4.9&interface=com.unj.dubbotest.provider.DemoService&methods=sayHello,getUsers&pid=8045&revision=0.0.1&side=provider&timestamp=1491546674441&version=0.0.

 

#serverMap.put(key, createServer(url)); key:10.199.66.242:20880 value:ExchangeServer

 

 

 

-> DubboProtocol createServer(URL url)

 

#返回HeaderExchangeServer,添加参数列表 如心跳,心跳时间

 

-> Exchangers.bind(url, requestHandler);

 

#返回HeaderExchangeServer,getTransporter()获取的实例来源于配置,默认返回一个NettyTransporter

 

-> HeaderExchangeServer.bind(URL url, ExchangeHandler handler);

 

 

 

-> HeaderExchangeServer(Transporters.bind(url, new DecodeHandler(new HeaderExchangeHandler(handler))));

 

#HeaderExchangeServer包装实例NettyServer

 

 

 

-> NettyTransporter.bind(URL url, ChannelHandler listener)

 

#return new NettyServer(url, listener)

 

 

 

-> NettyServer.doOpen();

 

#打开socket监听端口准备接收消息

 

#ServerBootstrap bind(getBindAddress())绑定地址端口

 

#RpcInvocation 具体类名、方法名、调用参数

 

#DubboInvoker – 执行具体的远程调用,包含初始化信息如client

 

#Protocol – 服务地址的发布和订阅

 

#Exporter – 暴露服务的引用,或取消暴露

 

 

 

二:consume(消费方):

 

->ReferenceConfig.init

 

#consume端启动初始化

 

->DubboProtocol.refer

 

#根据参数url,接口等构建Invoker

 

->JavassistProxyFactory.getProxy(Invoker<T> invoker, Class<?>[] interfaces)

 

#构建代理对象Proxy.getProxy(interfaces).newInstance(new InvokerInvocationHandler(invoker));

 

 

 

->DemoService.say(String hello);#真正调用时候

 

->InvokerInvocationHandler.invoke(Object proxy, Method method, Object[] args)

 

#invoker.invoke(new RpcInvocation(method, args)).recreate();RpcInvocation包装参数方法名

 

->DubboInvoker.doInovke(final Invocation invocation)

 

#统一代理调用

 

->ExchangeClient.send(invocation, isSent);

 

->HeaderExchangeChannel.request(Object request, int timeout)

 

->NettyChannel.send(Object message, boolean sent)

 

 

 

三:dubbo 底层通讯:NettyClient <-- 异步NIO传输 socket监听-> NettyServer

 

 

 

四:consume --> provider 调用过程:

 

-> NettyServer->NettyHandler.messageReceived #接收消息处理器

 

-> MultiMessageHandler->HeartbeatHandler->AllChannelHandler->DecodeHandler->HeaderExchangeHandler->DubboProtocol$requestHandler

 

#NettyServer启动时候绑定MultiMessageHandler

 

#DubboProtocol.getServers() 检索serverMap获取Exporter<?>

 

#DubboProtocol.getServers() 检索serverMap获取ExchangeServer

 

-> ExchangeHandlerAdapter.reply

 

#真正获取Invoker,将传入message 转换 invocation

 

-> invoker.invoke(invocation)

 

-> JavassistProxyFactory$AbstractProxyInvoker.doInvoke

 

#服务端Invoker代理 AbstractProxyInvoker调用目标引用service
1
1
分享到:
评论
2 楼 小楠人 2017-06-07  
同意一楼,看的太累
1 楼 somefuture 2017-06-06  
字太小了。。。。。。。。

相关推荐

    Dubbo调用java接口程序

    在实现Dubbo调用Java接口的过程中,我们需要以下步骤: 1. **创建服务接口**:定义一个Java接口,如`HelloService`,包含你需要的方法。 ```java public interface HelloService { String sayHello(String name);...

    jmeter测试dubbo请求

    而当我们谈论"jmeter测试dubbo请求"时,这意味着我们要利用JMeter对基于Dubbo框架的服务进行压力和负载测试。 Dubbo是阿里巴巴提供的一款高性能、轻量级的Java RPC框架,它使得服务调用变得简单,能有效提升分布式...

    dubbo调用的例子

    Dubbo调用例子详解 在分布式系统开发中,Dubbo是一个广泛应用的服务框架,它由阿里巴巴开源,旨在提高服务治理的效率和灵活性。本教程将深入探讨一个基于Dubbo的调用例子,涵盖从接口定义、服务提供者到消费者的...

    nodejs使用原生的dubbo协议打通了dubbo的rpc方法调用.

    标题中的“nodejs使用原生的dubbo协议打通了dubbo的rpc方法调用”意味着在Node.js环境中,开发者成功地实现了对Dubbo服务的RPC(远程过程调用)访问,利用了Dubbo协议的特性。Dubbo是阿里巴巴开源的一个高性能、轻量...

    Dubbo异步调用的优化共20页.pdf.zip

    【Dubbo异步调用的优化】是针对分布式服务框架Dubbo的一项重要技术,它能够显著提高系统的响应速度和处理能力。在高并发、大流量的业务场景下,异步调用是提升系统性能的关键手段之一。以下是关于Dubbo异步调用优化...

    Dubbo Dubbo

    由于Dubbo将这些协议的实现进行了封装了,无论是服务端(开发服务)还是客户端(调用服务),都不需要关心协议的细节,只需要在配置中指定使用的协议即可,从而保证了服务提供方与服务消费方之间的透明。 另外,如果...

    dubbo服务调用

    在本文中,我们将深入探讨如何利用 Dubbo 实现服务调用,并了解其超时、重试、服务降级和服务熔断等关键特性。 1. **服务调用** 在 Dubbo 中,服务调用是通过服务接口来完成的。服务提供者暴露服务接口,而服务...

    dubbo调用示例

    Dubbo的负载均衡策略(如随机、轮询、最少活跃调用数等)会在多个服务实例间分配请求,确保系统的高可用性。 此外,Dubbo还提供了监控、熔断、降级等高级特性,帮助开发者更好地管理和维护分布式系统。通过学习和...

    spring-cloud-alibaba+dubbo+nacos内部服务调用

    Dubbo的核心功能在于提供服务间的RPC(远程过程调用),使得服务调用如同本地方法调用一样简单。 而Nacos是阿里巴巴推出的另一个开源项目,它作为一个动态服务发现、配置管理和元数据共享中心,为微服务架构提供了...

    第四课:Dubbo调用模块详解1

    Dubbo的调用过程涉及到多个组件协同工作,主要包括: 1. **Invoker**:表示服务提供者的抽象,封装了实际的服务调用逻辑。 2. **Proxy**:动态生成的代理类,用于消费者调用。 3. **Registry**:服务注册中心,负责...

    dubbo源码分析系列

    - RPC(Remote Procedure Call)原理:深入解析Dubbo的RPC过程,包括请求构建、序列化、网络传输和反序列化响应等步骤。 - Netty通信框架:Dubbo如何利用Netty实现高效的网络通信,包括NIO模型、线程模型和缓冲区...

    dubbo分布式应用日志追踪

    在分布式系统中,尤其是采用了像Dubbo...总之,"dubbo分布式应用日志追踪"是解决分布式系统监控和故障排查的重要手段,通过结合Dubbo的SPI和MDC,我们可以有效地跟踪和分析服务间的调用,提升系统的可维护性和稳定性。

    dubbo捕获自定义异常_dubbo异常捕获_dubbo异常_自定义异常_捕捉异常_

    Dubbo 提供了强大的异常处理机制,使得服务提供者能够向消费者传递自定义异常,从而帮助消费者更好地理解和处理服务调用中的错误情况。本文将深入探讨如何在 Dubbo 中捕获自定义异常,以及相关的最佳实践。 1. **...

    整合dubbo、maven、spring、mybatis的服务提供者消费者调用错误示例

    在IT行业中,分布式服务框架Dubbo是阿里巴巴推出的一款高性能、轻量级的RPC(远程过程调用)框架,它能够使服务消费方和服务提供方通过网络进行通信。本示例中,我们将关注一个典型的问题——服务消费者在调用服务...

    Dubbo超时机制导致的雪崩连接

    根据给定的信息,“Dubbo超时机制导致的雪崩连接”主要指的是在Dubbo服务调用过程中,由于某种原因导致调用超时,进而使得大量请求堆积,最终导致整个系统无法正常提供服务的情况。具体来说: - **Dubbo服务调用...

    dubbo+zookeeper缓存方案

    2. **广播调用**:节点 A 在完成缓存更新后,会通过 Dubbo 的 Broadcast Cluster(广播调用)模式,向注册中心中的所有节点发送刷新缓存的请求。Broadcast Cluster 模式会依次调用所有提供者的接口,确保每个节点都...

    python 如何调用 dubbo 接口

    然而,有时我们需要在Python客户端应用中调用这些由Java编写并暴露为Dubbo协议的服务接口。Python语言本身并不直接支持Dubbo协议,因此我们需要使用一些间接的方法来实现跨语言的服务调用。 本篇将详细介绍如何使用...

    telnet dubbo接口测试工具

    【描述】"方便测试dubbo接口"表明这个工具简化了传统手动或编写代码进行Dubbo接口测试的过程,提高了开发效率。通常,测试Dubbo接口需要编写客户端代码,连接到服务提供者,发送请求并接收响应。此工具可能通过...

    dubbo的jar包 2.5.3和2.8.4

    - **高并发场景**:Dubbo的高性能特性使其在处理高并发请求时表现出色。 4. **如何使用** 使用Dubbo通常包括以下步骤: - **创建服务提供者**:编写服务接口和实现,配置服务暴露。 - **创建服务消费者**:引用...

    第四课:Dubbo调用模块详解 (2)1

    Dubbo的异步调用流程如下:调用者发送请求后,不会等待服务端的响应,而是立即返回,然后在后台处理结果,当结果准备就绪时,会通过回调或事件通知调用者。 总结来说,理解并掌握Dubbo调用模块的基本组成,尤其是...

Global site tag (gtag.js) - Google Analytics