前言


本篇文章讲述客户端与服务端的具体设计细节。有细心的小伙伴发现,客户端和服务端的工作方式不一样:服务端是多线程计算模型,利用工作线程完成数据的读取,而客户端是单线程(利用Reactor线程完成数据的读取)。这么做的原因有二:首先我们认为我们的使用RPC的初衷是由于CPU计算是瓶颈,不得已把计算放到多台机器上,所以服务端采用多线程计算模型;其次我们认为网络IO只要不是客户端故意阻塞,那么无论是请求数据还是响应数据只需要一次接收就可以收全,不会有线程长时间阻塞在网络上,所以客户端就使用反应器线程进行接收响应数据。

客户端同步和异步调用


SimpleRpc提供了同步调用和异步调用的方法,使用区别在于传递的参数不同,如下所示。

    //异步请求
int async_request(Server &server, Request *request, Response *response, ResultHandler *handler);
//同步请求
int sync_request(Server &server, Request *request, Response *response);

那么SimpleRpc对于同步和异步调用是如何支持的呢?我们重新看一下DownstreamHandler对数据的处理方式:

void DownstreamHandler::handle_read(int fd) {
char head[];
Connection conn(fd);
conn.recv_n(head, );
int size = *((int *)head);
char *buf = new char[size];
conn.recv_n(buf, size);
close(fd);
printf("Downstream Handler close fd:%d\n", fd);
//下游响应
_response->deserialize(buf, size);
//如果有result_handler,则调用data_comeback钩子函数
if(_result_handler != NULL) {
_result_handler->data_comeback(); //对于同步调用,这个方法会唤醒客户端使其从wait中返回
} delete[] buf;
//自杀
delete this;
}

result_handler的调用是关键,我们正是利用这一点做到同步调用和异步调用。ResultHandler的类UML如下:

DefaultResultHandler是SimpleRpc的默认结果处理方式,UserDefinedResultHandler由用户自己选择性的定义并实现。当客户端工作线程对服务端相应数据处理完毕后,调用ResultHandler的data_comeback方法执行这个钩子函数。

  • 同步调用的实现:
int SimpleRpcClient::sync_request(Server &server, Request *request, Response *response) {
Mutex mutex;
Connection conn;
Condition cond(&mutex);
InetAddr addr(server.get_port_str(), server.get_ip_str());
Connector conntor(addr);
int ret = conntor.Connect(conn); //建立与服务端的连接
if(ret == -){
LOG("connect error\n");
return -;
}
int size = request->bytes(); //获取请求序列化后的字节数
char *buf = new char[size + ]; //用额外4字节存放数据长度,方便接收端校验
if(buf == NULL) {
LOG("request oom, request need %d bytes\n", size + );
conn.Close();
return -;
}
int payload = request->serialize(buf + , size); //序列化
memcpy(buf, &payload, sizeof(int));
ret = conn.send_n(buf, payload + ); //发送序列化数据
if(ret != ) {
LOG("connection send error\n");
return -;
}
DefaultResultHandler *handler = new DefaultResultHandler(&cond, &mutex);
DownstreamHandler *down_handler =
new DownstreamHandler(conn.sock(), response, Reactor::get_instance(), handler);
Reactor::get_instance()->regist(conn.sock(), down_handler); //注册到reactor中等待响应事件的通知
handler->finish(); //阻塞调用,直到cond得到唤醒通知
delete[] buf;
delete handler;
return ;
}

我们的DefautlResultHandler拥有系统等待条件(Condition),并且作为DownstreamHandler的成员之一。客户端发送请求数据后,构造DownstreamHandler并注册到reactor中,等待服务端响应事件的通知。干完以上的事情之后,客户端应用线程调用DefaultResultHandler的finish方法阻塞直到得到完成通知,这样达到了同步调用的效果。

  • 异步调用的实现:

异步调用没有使用DefaultResultHandler作为参数传递给DownstreamHandler,而是把用户自定义的ResultHanlder传递进去,具体的控制流程(data_comeback函数)由用户自己定义。

int SimpleRpcClient::async_request(
Server &server, Request *request, Response *response, ResultHandler *handler) {
...
DownstreamHandler *down_handler =
new DownstreamHandler(conn.sock(), response, Reactor::get_instance(), handler);
Reactor::get_instance()->regist(conn.sock(), down_handler);
...
}

服务端工作线程计算模型


我们知道服务端使用多线程进行数据的处理,那么每个线程的工作内容是什么呢?

template<class REQUEST, class RESPONSE>
class Processor : public Worker<StreamEvent> {
public:
virtual int process(REQUEST &request, RESPONSE &response) = ; void run() {
while(true){
StreamEvent e = get_event(); //队列中获取待处理事件
char head[];
Connection conn(e.fd);
int payload = conn.recv_n(head, ); //接收数据长度
if(payload == -) {
close(e.fd);
printf("Error Processor close fd:%d\n", e.fd);
return;
} REQUEST request;
RESPONSE response; int size = *((int *)head);
char *recv_buf = new char[size];
conn.recv_n(recv_buf, size); //接收请求数据
request.deserialize(recv_buf, size); //反序列化 process(request, response); //进行用户代码逻辑计算,由用户实现
size = response.bytes();
char *send_buf = new char[size + ];
memcpy(send_buf, &size, sizeof(int));
payload = response.serialize(send_buf + , size); //序列化响应数据
conn.send_n(send_buf, size + ); //发送响应数据
//为了正常关闭该链接,需要重新注册回reactor
UpstreamHandler *upHandler = new UpstreamHandler(e.fd, Reactor::get_instance());
Reactor::get_instance()->regist(e.fd, upHandler);
delete recv_buf;
delete send_buf;
}
}
virutal ~Processor(){}
}

SimpleRpc-客户端与服务端工作模型探讨的更多相关文章

  1. Asp.Net MVC 模型验证详解-实现客户端、服务端双重验证

    概要 在asp.net webform开发中经常会对用户提交输入的信息进行校验,一般为了安全起见大家都会在客户端进行Javascript(利于交互).服务端双重校验(安全).书写校验代码是一个繁琐的过 ...

  2. NIO【同步非阻塞io模型】关于 NIO socket 的详细总结【Java客户端+Java服务端 + 业务层】【可以客户端间发消息】

    1.前言 以前使用 websocket来实现双向通信,如今深入了解了 NIO 同步非阻塞io模型 , 优势是 处理效率很高,吞吐量巨大,能很快处理大文件,不仅可以 做 文件io操作, 还可以做sock ...

  3. SignalR 实现web浏览器客户端与服务端的推送功能

    SignalR 是一个集成的客户端与服务器库,基于浏览器的客户端和基于 ASP.NET 的服务器组件可以借助它来进行双向多步对话. 换句话说,该对话可不受限制地进行单个无状态请求/响应数据交换:它将继 ...

  4. java客户端与服务端交互通用处理 框架解析

    一.综述 java 客户端与服务端交互过程中,采用NIO通讯是异步的,客户端基本采用同一处理范式,来进行同异步的调用处理. 处理模型有以下几个要素: 1. NIO发送消息后返回的Future 2. 每 ...

  5. 基于开源SuperSocket实现客户端和服务端通信项目实战

    一.课程介绍 本期带给大家分享的是基于SuperSocket的项目实战,阿笨在实际工作中遇到的真实业务场景,请跟随阿笨的视角去如何实现打通B/S与C/S网络通讯,如果您对本期的<基于开源Supe ...

  6. 在HTTP通讯过程中,是客户端还是服务端主动断开连接?

    比如说:IE访问IIS,获取文件,肯定是要建立一个连接,这个连接在完成通讯后,是客户端Close了连接,还是服务端Close了连接.我用程序测模拟IE和IIS,都没有收到断开连接的消息,也就是都没有触 ...

  7. 客户端与服务端的事件watcher源码阅读

    watcher存在的必要性 举个特容易懂的例子: 假如我的项目是基于dubbo+zookeeper搭建的分布式项目, 我有三个功能相同的服务提供者,用zookeeper当成注册中心,我的三个项目得注册 ...

  8. window安装rsync客户端和服务端

    原文地址: https://www.cnblogs.com/janas/p/3321087.html 下载地址: https://linux.linuxidc.com/index.php?folder ...

  9. TCP学习之五:客户端、服务端异步传输字符串

    参考学习张子阳大神的博客:http://www.cnblogs.com/JimmyZhang/category/101698.html 消息发送接口: 消息接收接口: 客户端: 服务端: 消息发送类: ...

随机推荐

  1. C#接口的简单创建及其用法

    我初次接触接口(Interface),对接口的作用有点迷茫,C#接口中包含方法.属性.索引器和事件的声明,但常用的接口中一般就是方法和属性,然而接口中并没有方法的具体实现代码(不能提供任何成员实现), ...

  2. 安装kafka过程及出现的问题解决

    第一步:下载kafka安装包 下载地址:http://kafka.apache.org/downloads 解压 到/usr/local 目录 tar -zxvf kafka_2.12-2.2.0 第 ...

  3. router-link 返回上页 和 新窗口打开链接

    1.如果使用了Vue-router的话,就可以用 this.$router.go(-1) 实现返回: 2.如果没使用vue-router,就可以用 window.history.go(-1) 实现返回 ...

  4. Java软件工程的弹幕调试原则

    日期:2019.4.25 博客期:054 星期四 今天是把很久之前的那个相关程序——一维数组的最大和的子数组的求取信息,我们今天的任务就是把每一步的信息都要进行输出查看! 如下图: package p ...

  5. PHP-高并发和大流量的解决方案

    一  高并发的概念在互联网时代,并发,高并发通常是指并发访问.也就是在某个时间点,有多少个访问同时到来. 二  高并发架构相关概念1.QPS (每秒查询率) : 每秒钟请求或者查询的数量,在互联网领域 ...

  6. 物化视图(materialized view) 实现数据迁移、数据定时同步

    近日公司有一个9i 的Oracle数据库,运行效率低下.想要将其升级到11G. 但是升级之前 要将数据进行同步,好在表不是很多.只有三张表.业务压力也不大,就想到了使用物 化视图的方式将数据同步过来. ...

  7. 逻辑卷管理(linux)

    (创建分区)1.fdisk /dev/sdb2.n3default4.default5.+200M6.w7.partprobe(更新分区)8mkfs.ext4 /dev/sdb1//格式化sdb1.. ...

  8. Zabbix (五)

    介绍添加主机时,各个参数的含义 https://blog.51cto.com/5001660/2154692 Zabbix配置介绍: https://blog.51cto.com/5001660/21 ...

  9. PHP字符过滤方法

    function str_filter_replace($str) { if (empty($str)) return false; $str = htmlspecialchars($str); $s ...

  10. Java 多线程系列 CountDownLatch

    CountDownLatch 一个或多个线程等待其他线程完成操作后在在执行 CountDownLatch通过一个计数器来实现,await方法阻塞直到 countDown() 调用计数器归零之后释放所有 ...