- 浏览: 77708 次
文章分类
最新评论
-
kevinflynn:
...
ThreadLocal 源码分析 -
kevinflynn:
[url=aaaa][/url]
ThreadLocal 源码分析 -
kevinflynn:
学习到了 感谢楼主。
ThreadLocal 源码分析
remote 层关注 transport 和 exchange 这两个包即可. 因为这两个包下封装了通讯相关的内容.
先来补充点额外知识:
1.Endpoint 是端点的概念,我们可以看到,对其抽象,最主要的是一个 send 方法,用于发送数据. 但是我有一个疑问?为啥 Endpoint 不抽象出 receive ?
2.Endpoint 的 send 方法和 ChannelHandler 的 sent 方法的区别和联系?
刚刚我看了下 netty4 下面的 NettyServerHandler 和 NettyChannel,发现 ChannelHandler 中定义的 sent 方法、receive 方法和 Netty Channel 的实现有关,而 NettyChannel 中的 send 方法是和数据发送有关的.
换句话说:NettyChannel 中的 send 方法在发送的时候,会调用 ChannelHandler 中定义的对应方法进行处理.
其实上面问的 Endpoint 为啥不抽象出 receive?其实这个问题就很怪异. 因为 send 方法是一个主动的过程,我发送一个消息,主动的调用下就好,但是我收到一个消息,就不是主动调用下就可以完成的。或许有人会说,可以向 socket 编程那样阻塞住啊。但是仔细想想,netty 不是有 handler 可以在收到消息时进行处理吗?所以在 Endpoint 接口中抽象 receive 方法很鸡肋.
channel 是通讯的载体,信息通过 channel 进行发送和接收. channel 继承 Endpoint. 同时定义了和 attribute 相关的方法.
channelHandler 和 Netty 中的 channelHandler 相关联,是对 channel 的逻辑处理,同时该接口是一个可扩展接口.
client 继承 Endpoint,Channel,Resetable,IdleSensible 接口,客户端同样是一个端点,需要注意的是,客户端可以重连服务端.
server 继承 Endpoint,Resetable,IdleSensible 接口,可以返回所有连接到该服务端的所有客户端,同时可以根据 ip 地址获取具体的客户端.
Codec2 是编解码类
Decodeable 解码接口,用来判断消息是否可以被解码等
Dispatcher 接口用于调度,选择不同的 handler 来进行处理(将消息派发到线程池进行处理).
Transport 接口用于获取 Client 和 Server.
先去网上盗一张图.
下面我们按照上图进行分析:
AbstractPeer 实现 Endpoint 和 ChannelHandler. 主要是增加了对通道是否关闭的判断.
AbstractEndpoint 继承 AbstractPeer 类,增加了 codec2,超时时间,连接超时时间等属性.
AbstractServer 类继承 AbstractEndpoint 以及实现 Server 接口. 这在一方面印证了 Server 其实是一个端.
1.AbstractServer 中的 executor 是在哪个地方用到的了?首先我们看到在构造方法中看到其从 dataStore 中获取 executor. 那么是在哪里放进去的了?我们可以明确的从 WrappedChannelHandler 类的构造方法中看到这里对 executor 进行了赋值,如下:
public WrappedChannelHandler(ChannelHandler handler, URL url) {
this.handler = handler;
this.url = url;
executor = (ExecutorService) ExtensionLoader.getExtensionLoader(ThreadPool.class).getAdaptiveExtension().getExecutor(url);
String componentKey = Constants.EXECUTOR_SERVICE_COMPONENT_KEY;
if (CONSUMER_SIDE.equalsIgnoreCase(url.getParameter(SIDE_KEY))) {
componentKey = CONSUMER_SIDE;
}
DataStore dataStore = ExtensionLoader.getExtensionLoader(DataStore.class).getDefaultExtension();
dataStore.put(componentKey, Integer.toString(url.getPort()), executor);
}
2.现在分析下这里的 executor 干了个啥事了?优雅关机.
优雅停机是啥意思了?核心业务在服务器中正在执行时中断,可能出现严重后果,所以需要等待服务执行完后关闭.
dubbo 是如何做的了?
1.收到 kill 9 进程退出信号,Spring 容器会出发容器销毁事件
2.privoder 端会取消服务元数据信息
3.consumer 端会收到最新地址列表(不包含准备停机的地址)
4.Dubbo 协议会发送 readonly 事件报文通知 consumer 服务不可用
5.服务端等待已经执行的任务结束并拒绝新任务执行
AbstractServer 最主要是实现了 send 和 close 方法. 同时值得注意的是该构造方法调用了 doOpen 方法打开一个服务端.
AbstractClient 继承 AbstractEndpoint 并实现 Client 接口,也就是说 AbstractClient 在语义上是一个端. 同时值得注意的是该构造方法调用了 doOpen 方法和 connect 方法,完成客户端的初始化工作. 我们需要注意下,AbstractClient 中的 wrapChannelHandler 对 handler 的语义进行了增强,增加了心跳和多消息处理 handler. AbstractClient 的 send 方法和 AbstractServer 的 send 方法在实现的时候是不同的,Server 端的 send 方法是遍历 channel,然后调用每一个 channel 进行发送. Client 端只是调用 channel.send 进行发送.
connect 方法加锁了,避免了多线程同时对一个 channel 调用 connect 方法. Client 主要实现 connect、disconnect、reconnect 及 close 的功能.
AbstractChannel 继承 AbstractPeer,实现 Channel 接口. Channel 在 dubbo 看来,也是一个 Endpoint. 是数据的载体,客户端和服务端通过 channel 发送和接收数据,需要注意的是:AbstractPeer 实现了 ChannelHandler 接口.
ChannelHandlerDelegate 作为装饰者模式的 component. 该接口只定义了一个 getHandler 方法.
AbstractChannelHandlerDelegate 装饰角色,实现的方法都是直接调用成员变量 handler 中的方法.
DecodeHandler 继承 AbstractChannelHandlerDelegate,重写了 handler 的 receive 方法. 在接收到消息的时候进行了判断,如果消息可解码,则直接调用 decode 方法,如果消息是 request 类型的,则对数据部分进行解码,如果消息是 response 类型的,则对结果进行解码. 解码完成后,向上进行传递,换句话说可以无限对 handler 的功能进行增强. 其实这个设计和责任链的设计是两种思路吧,一种是责任链的方式,一种是装饰者模式(任何被解码的消息都必须可以被解码,即:实现 Decodeable 接口).
MultiMessageHandler 同样实现 AbstractChannelHandlerDelegate 接口,但是可以处理 MultiMessage. 我们可以把 MultiMessage 看成是一个聚合的消息(ArrayList). 所以处理的时候也是简单粗暴,直接遍历,然后调用 received 方法.
WrappedChannelHandler 同样是装饰者角色,同样是直接调用成员变量 handler 中的方法. 但是值得注意的是:该类增加了线程池,它的子类都是关心那些消息需要交给线程池处理,那些消息直接交由 I/O 线程处理.
public WrappedChannelHandler(ChannelHandler handler, URL url) {
this.handler = handler;
this.url = url;
executor = (ExecutorService) ExtensionLoader.getExtensionLoader(ThreadPool.class).getAdaptiveExtension().getExecutor(url);
String componentKey = Constants.EXECUTOR_SERVICE_COMPONENT_KEY;
if (CONSUMER_SIDE.equalsIgnoreCase(url.getParameter(SIDE_KEY))) {
componentKey = CONSUMER_SIDE;
}
DataStore dataStore = ExtensionLoader.getExtensionLoader(DataStore.class).getDefaultExtension();
dataStore.put(componentKey, Integer.toString(url.getPort()), executor);
}
MessageOnlyChannelHandler 只有请求响应消息分发到线程池,其他连接断开、心跳等事件在 I/O 线程上执行. MessageOnlyChannelHandler 在重写 received 方法时,并没有 AllChannelHandler 中的那样处理,为什么了?因为该类抛出的异常,会在 HeaderExchangeHandler.caught 方法中进行统一处理.
ExecutionChannelHandler 只有请求消息派发到线程池(不含响应),响应、断开连接、心跳等事件在 I/O 线程上执行. 但是我们看到这里执行的时候,又和 MessageOnlyChannelHandler 中的处理方式不同?为什么了?因为 MessageOnlyChannelHandler 中不仅要处理 request,还要处理 response. 所以进行了统一的处理,但是 ExecutionChannelHandler 只需要在线程池中处理 request,所以就可以和 AllDispatcher 一样了(不必须).
AllChannelHandler 所有消息都派发到线程池,包括请求、响应、连接事件、断开事件、心跳等事件. 关于使用 AllDispatcher 的时候,在早期版本 dubbo provider 线程池可能被 exhausted. 这是由于异常也需要使用线程池来进行处理,但是线程池还是满的,所以无法进一步处理,consumer 只能死等,直至超时. 如何规避了?我们看到在 dubbo2.7 中,received 方法收到消息后,如果线程池抛出了拒绝异常,且这是一个请求,那么将会收到服务端线程池被打满的消息.
direct 所有消息都不派发到线程池,直接在 I/O 线程上执行.
ConnectionOrderedChannelHandler 在 I/O 线程上,将连接、断开事件放入队列,有序逐个执行,其他事件派发到线程池. 需要说一下的是:该类有自己的线程池 connectionExecutor. 因为该类的 caught 方法同样将任务分给了线程池,所以其 received 方法,必须要进行判断处理,如下:
public void received(Channel channel, Object message) throws RemotingException {
ExecutorService executor = getExecutorService();
try {
executor.execute(new ChannelEventRunnable(channel, handler, ChannelState.RECEIVED, message));
} catch (Throwable t) {
//fix, reject exception can not be sent to consumer because thread pool is full, resulting in consumers waiting till timeout.
if (message instanceof Request && t instanceof RejectedExecutionException) {
Request request = (Request) message;
if (request.isTwoWay()) {
String msg = "Server side(" + url.getIp() + "," + url.getPort() + ") threadpool is exhausted ,detail msg:" + t.getMessage();
Response response = new Response(request.getId(), request.getVersion());
response.setStatus(Response.SERVER_THREADPOOL_EXHAUSTED_ERROR);
response.setErrorMessage(msg);
channel.send(response);
return;
}
}
throw new ExecutionException(message, channel, getClass() + " error when process received event .", t);
}
}
Dispatcher 就不说了,直接 new 对应的 handler.
ChannelHandlers 类是对 Dispatcher 返回的 channelHandler 的功能进行了增强,增加了处理心跳和多消息的 handler.
AbstractCodec 实现 Codec2 接口,主要定义了检查数据长度方法、判断是客户端侧,还是服务端侧方法,获取序列化方式方法.
TransportCodec 继承 AbstractCodec,主要对 encode 方法、decode 方法进行了重写. 实现序列化和反序列化的功能.
CodecAdapter 将 Codec 适配为 Codec2.
ChannelDelegate 实现 Channel 接口,为 Channel 的装饰角色.
ClientDelegate 实现 Client 接口,为 Client 的装饰角色.
ServerDelegate 实现 Server 接口,为 Server 的装饰角色.
ChannelHandlerAdapter 实现 ChannelHandler 接口,实现的都是空方法,方便子类重写.
ChannelHandlerDispatcher 为 ChannelHandler 的调度器,该类有一个成员变量 channelHandlers,用于保存 channelHandlers. 在实现 connected、disconnected 等方法的时候,都会遍历这个属性列表,依次调用.
CodecSupport 编解码工具类,用于查询序列化方式.
参考:
① https://segmentfault.com/a/1190000017390253#articleHeader28
先来补充点额外知识:
1.Endpoint 是端点的概念,我们可以看到,对其抽象,最主要的是一个 send 方法,用于发送数据. 但是我有一个疑问?为啥 Endpoint 不抽象出 receive ?
2.Endpoint 的 send 方法和 ChannelHandler 的 sent 方法的区别和联系?
刚刚我看了下 netty4 下面的 NettyServerHandler 和 NettyChannel,发现 ChannelHandler 中定义的 sent 方法、receive 方法和 Netty Channel 的实现有关,而 NettyChannel 中的 send 方法是和数据发送有关的.
换句话说:NettyChannel 中的 send 方法在发送的时候,会调用 ChannelHandler 中定义的对应方法进行处理.
其实上面问的 Endpoint 为啥不抽象出 receive?其实这个问题就很怪异. 因为 send 方法是一个主动的过程,我发送一个消息,主动的调用下就好,但是我收到一个消息,就不是主动调用下就可以完成的。或许有人会说,可以向 socket 编程那样阻塞住啊。但是仔细想想,netty 不是有 handler 可以在收到消息时进行处理吗?所以在 Endpoint 接口中抽象 receive 方法很鸡肋.
channel 是通讯的载体,信息通过 channel 进行发送和接收. channel 继承 Endpoint. 同时定义了和 attribute 相关的方法.
channelHandler 和 Netty 中的 channelHandler 相关联,是对 channel 的逻辑处理,同时该接口是一个可扩展接口.
client 继承 Endpoint,Channel,Resetable,IdleSensible 接口,客户端同样是一个端点,需要注意的是,客户端可以重连服务端.
server 继承 Endpoint,Resetable,IdleSensible 接口,可以返回所有连接到该服务端的所有客户端,同时可以根据 ip 地址获取具体的客户端.
Codec2 是编解码类
Decodeable 解码接口,用来判断消息是否可以被解码等
Dispatcher 接口用于调度,选择不同的 handler 来进行处理(将消息派发到线程池进行处理).
Transport 接口用于获取 Client 和 Server.
先去网上盗一张图.
下面我们按照上图进行分析:
AbstractPeer 实现 Endpoint 和 ChannelHandler. 主要是增加了对通道是否关闭的判断.
AbstractEndpoint 继承 AbstractPeer 类,增加了 codec2,超时时间,连接超时时间等属性.
AbstractServer 类继承 AbstractEndpoint 以及实现 Server 接口. 这在一方面印证了 Server 其实是一个端.
1.AbstractServer 中的 executor 是在哪个地方用到的了?首先我们看到在构造方法中看到其从 dataStore 中获取 executor. 那么是在哪里放进去的了?我们可以明确的从 WrappedChannelHandler 类的构造方法中看到这里对 executor 进行了赋值,如下:
public WrappedChannelHandler(ChannelHandler handler, URL url) {
this.handler = handler;
this.url = url;
executor = (ExecutorService) ExtensionLoader.getExtensionLoader(ThreadPool.class).getAdaptiveExtension().getExecutor(url);
String componentKey = Constants.EXECUTOR_SERVICE_COMPONENT_KEY;
if (CONSUMER_SIDE.equalsIgnoreCase(url.getParameter(SIDE_KEY))) {
componentKey = CONSUMER_SIDE;
}
DataStore dataStore = ExtensionLoader.getExtensionLoader(DataStore.class).getDefaultExtension();
dataStore.put(componentKey, Integer.toString(url.getPort()), executor);
}
2.现在分析下这里的 executor 干了个啥事了?优雅关机.
优雅停机是啥意思了?核心业务在服务器中正在执行时中断,可能出现严重后果,所以需要等待服务执行完后关闭.
dubbo 是如何做的了?
1.收到 kill 9 进程退出信号,Spring 容器会出发容器销毁事件
2.privoder 端会取消服务元数据信息
3.consumer 端会收到最新地址列表(不包含准备停机的地址)
4.Dubbo 协议会发送 readonly 事件报文通知 consumer 服务不可用
5.服务端等待已经执行的任务结束并拒绝新任务执行
AbstractServer 最主要是实现了 send 和 close 方法. 同时值得注意的是该构造方法调用了 doOpen 方法打开一个服务端.
AbstractClient 继承 AbstractEndpoint 并实现 Client 接口,也就是说 AbstractClient 在语义上是一个端. 同时值得注意的是该构造方法调用了 doOpen 方法和 connect 方法,完成客户端的初始化工作. 我们需要注意下,AbstractClient 中的 wrapChannelHandler 对 handler 的语义进行了增强,增加了心跳和多消息处理 handler. AbstractClient 的 send 方法和 AbstractServer 的 send 方法在实现的时候是不同的,Server 端的 send 方法是遍历 channel,然后调用每一个 channel 进行发送. Client 端只是调用 channel.send 进行发送.
connect 方法加锁了,避免了多线程同时对一个 channel 调用 connect 方法. Client 主要实现 connect、disconnect、reconnect 及 close 的功能.
AbstractChannel 继承 AbstractPeer,实现 Channel 接口. Channel 在 dubbo 看来,也是一个 Endpoint. 是数据的载体,客户端和服务端通过 channel 发送和接收数据,需要注意的是:AbstractPeer 实现了 ChannelHandler 接口.
ChannelHandlerDelegate 作为装饰者模式的 component. 该接口只定义了一个 getHandler 方法.
AbstractChannelHandlerDelegate 装饰角色,实现的方法都是直接调用成员变量 handler 中的方法.
DecodeHandler 继承 AbstractChannelHandlerDelegate,重写了 handler 的 receive 方法. 在接收到消息的时候进行了判断,如果消息可解码,则直接调用 decode 方法,如果消息是 request 类型的,则对数据部分进行解码,如果消息是 response 类型的,则对结果进行解码. 解码完成后,向上进行传递,换句话说可以无限对 handler 的功能进行增强. 其实这个设计和责任链的设计是两种思路吧,一种是责任链的方式,一种是装饰者模式(任何被解码的消息都必须可以被解码,即:实现 Decodeable 接口).
MultiMessageHandler 同样实现 AbstractChannelHandlerDelegate 接口,但是可以处理 MultiMessage. 我们可以把 MultiMessage 看成是一个聚合的消息(ArrayList). 所以处理的时候也是简单粗暴,直接遍历,然后调用 received 方法.
WrappedChannelHandler 同样是装饰者角色,同样是直接调用成员变量 handler 中的方法. 但是值得注意的是:该类增加了线程池,它的子类都是关心那些消息需要交给线程池处理,那些消息直接交由 I/O 线程处理.
public WrappedChannelHandler(ChannelHandler handler, URL url) {
this.handler = handler;
this.url = url;
executor = (ExecutorService) ExtensionLoader.getExtensionLoader(ThreadPool.class).getAdaptiveExtension().getExecutor(url);
String componentKey = Constants.EXECUTOR_SERVICE_COMPONENT_KEY;
if (CONSUMER_SIDE.equalsIgnoreCase(url.getParameter(SIDE_KEY))) {
componentKey = CONSUMER_SIDE;
}
DataStore dataStore = ExtensionLoader.getExtensionLoader(DataStore.class).getDefaultExtension();
dataStore.put(componentKey, Integer.toString(url.getPort()), executor);
}
MessageOnlyChannelHandler 只有请求响应消息分发到线程池,其他连接断开、心跳等事件在 I/O 线程上执行. MessageOnlyChannelHandler 在重写 received 方法时,并没有 AllChannelHandler 中的那样处理,为什么了?因为该类抛出的异常,会在 HeaderExchangeHandler.caught 方法中进行统一处理.
ExecutionChannelHandler 只有请求消息派发到线程池(不含响应),响应、断开连接、心跳等事件在 I/O 线程上执行. 但是我们看到这里执行的时候,又和 MessageOnlyChannelHandler 中的处理方式不同?为什么了?因为 MessageOnlyChannelHandler 中不仅要处理 request,还要处理 response. 所以进行了统一的处理,但是 ExecutionChannelHandler 只需要在线程池中处理 request,所以就可以和 AllDispatcher 一样了(不必须).
AllChannelHandler 所有消息都派发到线程池,包括请求、响应、连接事件、断开事件、心跳等事件. 关于使用 AllDispatcher 的时候,在早期版本 dubbo provider 线程池可能被 exhausted. 这是由于异常也需要使用线程池来进行处理,但是线程池还是满的,所以无法进一步处理,consumer 只能死等,直至超时. 如何规避了?我们看到在 dubbo2.7 中,received 方法收到消息后,如果线程池抛出了拒绝异常,且这是一个请求,那么将会收到服务端线程池被打满的消息.
direct 所有消息都不派发到线程池,直接在 I/O 线程上执行.
ConnectionOrderedChannelHandler 在 I/O 线程上,将连接、断开事件放入队列,有序逐个执行,其他事件派发到线程池. 需要说一下的是:该类有自己的线程池 connectionExecutor. 因为该类的 caught 方法同样将任务分给了线程池,所以其 received 方法,必须要进行判断处理,如下:
public void received(Channel channel, Object message) throws RemotingException {
ExecutorService executor = getExecutorService();
try {
executor.execute(new ChannelEventRunnable(channel, handler, ChannelState.RECEIVED, message));
} catch (Throwable t) {
//fix, reject exception can not be sent to consumer because thread pool is full, resulting in consumers waiting till timeout.
if (message instanceof Request && t instanceof RejectedExecutionException) {
Request request = (Request) message;
if (request.isTwoWay()) {
String msg = "Server side(" + url.getIp() + "," + url.getPort() + ") threadpool is exhausted ,detail msg:" + t.getMessage();
Response response = new Response(request.getId(), request.getVersion());
response.setStatus(Response.SERVER_THREADPOOL_EXHAUSTED_ERROR);
response.setErrorMessage(msg);
channel.send(response);
return;
}
}
throw new ExecutionException(message, channel, getClass() + " error when process received event .", t);
}
}
Dispatcher 就不说了,直接 new 对应的 handler.
ChannelHandlers 类是对 Dispatcher 返回的 channelHandler 的功能进行了增强,增加了处理心跳和多消息的 handler.
AbstractCodec 实现 Codec2 接口,主要定义了检查数据长度方法、判断是客户端侧,还是服务端侧方法,获取序列化方式方法.
TransportCodec 继承 AbstractCodec,主要对 encode 方法、decode 方法进行了重写. 实现序列化和反序列化的功能.
CodecAdapter 将 Codec 适配为 Codec2.
ChannelDelegate 实现 Channel 接口,为 Channel 的装饰角色.
ClientDelegate 实现 Client 接口,为 Client 的装饰角色.
ServerDelegate 实现 Server 接口,为 Server 的装饰角色.
ChannelHandlerAdapter 实现 ChannelHandler 接口,实现的都是空方法,方便子类重写.
ChannelHandlerDispatcher 为 ChannelHandler 的调度器,该类有一个成员变量 channelHandlers,用于保存 channelHandlers. 在实现 connected、disconnected 等方法的时候,都会遍历这个属性列表,依次调用.
CodecSupport 编解码工具类,用于查询序列化方式.
参考:
① https://segmentfault.com/a/1190000017390253#articleHeader28
发表评论
-
dubbo-common-rpc
2020-12-30 22:46 403ApplicationModel 实例模型. ... -
dubbo-filter 源码分析
2019-12-01 23:08 466说明:这个模块的功能是缓存调用结果,但是我有一个疑问,比如说开 ... -
dubbo 配置中心源码分析(一)
2019-11-28 23:09 533Configuration 接口分析 /** * 配置中心 ... -
dubbo 监控中心源码分析
2019-11-28 00:15 534MonitorFactory 创建 Monitor 的接口 ... -
dubbo injvm 协议分析
2019-11-21 23:49 811在看 injvm 协议的时候,我们还是从 InjvmProto ... -
[dubbo] resteasy 调用流程分析
2019-11-21 00:46 740在看 dubbo rest 协议的时候,很多东西都不是很懂,特 ... -
【dubbo】基于httpclient实现rpc的思路
2019-11-18 22:05 720今天在看 dubbo http 协议的时候,发现它是使用 js ... -
dubbo http 协议分析
2019-11-13 23:30 678今天看 dubbo http 协议的时候,觉得很好玩,开始的理 ... -
dubbo remoting 层之 exchange
2019-10-31 00:38 623补充点额外知识: CompletableFuture 现在 ... -
dubbo 序列化是如何封装的
2019-10-27 17:55 553dubbo 是支持多种序列化方式的,那么它就必须在此基础上做一 ... -
dubbo 线程池
2019-08-31 23:18 0dubbo 中有四个线程池: CachedThreadPoo ... -
dubbo 线程池
2019-08-31 23:18 769dubbo 中有四个线程池: CachedThreadPoo ... -
关于 dubbo 的 SpiExtensionFactory 的一点说明
2019-08-29 00:30 672关于 dubbo 的 SpiExtensionFa ...
相关推荐
2. **Remoting层解析**:Dubbo的Remoting层是其通信基础,涉及到网络通信、序列化和协议处理等关键技术。书中详细解析了Netty、HTTP、TCP等通信机制,以及Hessian和Protobuf等序列化方式,让读者能够理解服务间的...
1. `RemotingServer`:这是服务端的核心接口,它负责接收客户端的连接请求,管理与客户端的连接,并将接收到的请求转发给上层的服务处理层。 2. `RemotingClient`:客户端的实现,它负责建立到服务端的连接,发送...
在Dubbo项目中,Remoting层负责服务之间的远程调用,它需要一个高效的通信框架来支持。Netty4因其优异的性能和易用性,被选为Dubbo的默认网络通信引擎。Dubbo-Remoting-Netty4模块就是基于Netty4实现的远程通信组件...
如果使用dubbo遇到错误:com.alibaba.dubbo.remoting.RemotingException: Fail to decode request due to: RpcInvocation 请下载这个jar,替换掉你项目中的那个jar,应该可以解决。
dubbo-remoting-grizzly dubbo-remoting-http dubbo-remoting-mina dubbo-remoting-netty dubbo-remoting-p2p dubbo-remoting-zookeeper dubbo-rpc dubbo-rpc-api dubbo-rpc-default dubbo-rpc-hessian dubbo-rpc-...
Dubbo是中国阿里巴巴开源的...通过深入分析Dubbo的源码,开发者可以更好地理解其内部机制,从而优化服务性能,定制更符合业务需求的解决方案。同时,源码分析也是提升自身技术能力,掌握分布式系统设计原理的重要途径。
at com.alibaba.dubbo.remoting.exchange.support.DefaultFuture.returnFromResponse(DefaultFuture.java:190) at com.alibaba.dubbo.remoting.exchange.support.DefaultFuture.get(DefaultFuture.java:110) at ...
综上所述,通过对Dubbo源码的详细分析,我们可以了解到它不仅仅是一个简单的RPC框架,而是涵盖了服务治理、集群容错、负载均衡等多个方面的复杂系统。希望以上解析能够帮助读者更深入地理解Dubbo的内部工作原理和...
- **dubbo-remoting-api**:远程通信模块,抽象了网络通讯的核心接口,如传输器、客户端、服务器、编码解码器、数据交换等。提供了Netty、Mina、Grizzly、HTTP、P2P和Zookeeper等多种实现,实现远程方法调用。 - **...
1. 远程调用(RPC):Dubbo的Remoting层处理服务间的网络通信,包括协议解析、序列化、连接管理等。Protocol接口定义了服务调用的基本操作,如refer、invoke等,而Exporter和Invoker接口则封装了服务暴露和调用的...
[INFO] dubbo-remoting .................................... SUCCESS [0.151s] [INFO] dubbo-remoting-api ................................ SUCCESS [6.705s] [INFO] dubbo-remoting-netty .......................
5. **源文件概述**:Dubbo的源代码包括多个模块,如dubbo-common、dubbo-remoting、dubbo-rpc、dubbo-cluster等,分别处理基础工具、网络通信、远程调用、集群逻辑等。 6. **核心机制分析**: - **服务注册与发现*...
书中深入剖析了Dubbo的内部实现原理,如Remoting通信层的设计,包括Netty、MinAIO等网络库的使用,以及RPC协议的解析和处理流程。此外,还会涉及服务治理的高级特性,如动态配置、服务路由规则、集群策略(如广播、...
处理交换层的请求与响应,`com.alibaba.dubbo.remoting.Transporter`负责传输层的创建。 4.2 序列化与反序列化 `com.alibaba.dubbo.common.serialize.Serializer`接口定义了数据的序列化与反序列化,Dubbo内置了...
《淘宝Dubbo框架详解》 在当今的互联网行业中,服务化的架构已经成为主流,而作为阿里巴巴开源的一款高性能、轻量级的分布式服务框架,淘宝Dubbo无疑是服务化领域的一颗璀璨明珠。本文将深入探讨Dubbo的核心特性、...
java运行依赖jar包
java运行依赖jar包
java运行依赖jar包
java运行依赖jar包
java运行依赖jar包