智能 RPC框架 (C++)
RPC中文叫远程函数调用,它是一种通信方式,只是看起来像普通的函数调用。
它包括三个基本要素:
1:服务端注册相应的(服务)函数(用于调用方调用)
2:调用方通过函数调用的方式将一些信息和参数打包到消息,然后发送消息给被调用方。
3:被调用方收到消息后,提取信息和参数。调用相应函数。
被调用方不需要用户手动解析参数,而是由"包装代码"预先解析出来。
目前很多rpc框架都(设计)配有协议描述文件,通过代码生成,产生((含有)"包装代码")服务端的服务类或函数。
我不喜欢代码生成,我喜欢直接在代码中搞定它。
果然,我最近看到有朋友在一些脚本语言中做到这点。某些实现还不需要手动(预先)注册服务函数。
比如:
https://github.com/sniperHW/distri.lua/blob/master/examples/rpcserver.lua
https://github.com/akirayu101/GM_RPC/blob/master/gmrpc_lua/rpc_handlers/rpc_handler_sample.lua
https://github.com/akirayu101/GM_RPC/blob/master/gmrpc_py/rpc_handlers/rpc_handler_sample.py
然而我又不熟悉lua或python,所以我用C++11 来实现了它。
主要功能:
1:注册服务函数
void test5(string a, int b, map<int, map<int, string>> vlist)
{
} rpc rpc_server; /*rpc服务器*/
rpc_server.def("test5", test5);
2:客户端调用远程函数
rpc rpc_client; /*rpc客户端*/
rpc_client.call("test5", "a", 1, mlist, [&upvalue](int a, int b){
upvalue++;
cout << "upvalue:" << upvalue << ", a:" << a << ", b:" << b << endl;
});
其中mlis是一个map<int,map<int,string>>类型变量。 rpc_client.call 的返回值是一个string,它表示此次call的消息。
我们可以把它(string 消息)通过网络发送给服务器。在这里(测试)我们直接通过下面的方式传递给服务端。
!!!注意!!!:call的最后一个参数可以是一个lambda,它表示处理此rpc返回值。 如果不是一个lambda,则它也是rpc调用参数。
3:服务端处理rpc request
rpc_server.handleRpc(rpc_request_msg);
其中 rpc_request_msg为接受到的网络消息(字符串)。
这样就会自动调用到我们的 test5 函数。 并且形参已经(自动)准备OK。你只需要在test5 里使用这些参数即可。(不用关心网络消息协议)。
4:被调用方可以返回数据给调用方
rpc_response_str = rpc_server.reply(1, 1, 2); /* (1,1,2)中的1为调用方的req_id, (1,2)为返回值 */
rpc_client.handleResponse(rpc_response_str);
上面代码通过 rpc_server.reply返回消息给客户端。 然后客户端模拟收到消息后通过 rpc_client.handleResponse(rpc_response_str)
会回调rpc_client.call() 时 所传递的lambda回调函数。
注意:以上 服务函数(譬如test5)和rpc 返回值处理函数(譬如那个lambda)的参数 是任意个数,且"任意"类型
(支持 int,string,JsonObject-json对象,vector<int>,vector<string>, map<int,string>,map<string,int>,map<string,string>, map<int/string, 前述所有类型/递归> )
整个测试代码:
void test1(int a, int b)
{
cout << "in test1" << endl;
cout << a << ", " << b << endl;
} void test2(int a, int b, string c)
{
cout << "in test2" << endl;
cout << a << ", " << b << ", " << c << endl;
} void test3(string a, int b, string c)
{
cout << "in test3" << endl;
cout << a << ", " << b << ", " << c << endl;
} void test4(string a, int b)
{
cout << "in test4" << endl;
cout << a << "," << b << endl;
} void test5(string a, int b, map<int, map<int, string>> vlist)
{
} void test6(string a, int b, map<string, int> vlist)
{
} void test7()
{
cout << "in test7" << endl;
} int main()
{
int upvalue = 10;
using namespace dodo; rpc rpc_server; /*rpc服务器*/
rpc rpc_client; /*rpc客户端*/ rpc_server.def("test4", test4);
rpc_server.def("test5", test5);
rpc_server.def("test7", test7); string rpc_request_msg; /* rpc消息 */
string rpc_response_str; /* rpc返回值 */ {
rpc_request_msg = rpc_client.call("test7"); rpc_server.handleRpc(rpc_request_msg);
} map<int, string> m1;
m1[1] = "Li";
map<int, string> m2;
m2[2] = "Deng";
map<int, map<int, string>> mlist;
mlist[100] = m1;
mlist[200] = m2; {
rpc_request_msg = rpc_client.call("test5", "a", 1, mlist, [&upvalue](int a, int b){
upvalue++;
cout << "upvalue:" << upvalue << ", a:" << a << ", b:" << b << endl;
}); rpc_server.handleRpc(rpc_request_msg);
} {
rpc_request_msg = rpc_client.call("test5", "a", 1, mlist, [&upvalue](string a, string b, int c){
upvalue++;
cout << "upvalue:" << upvalue << ", a:" << a << ", b:" << b << ", c:" << c << endl;
}); rpc_server.handleRpc(rpc_request_msg);
} {
rpc_request_msg = rpc_client.call("test4", "a", 1);
rpc_server.handleRpc(rpc_request_msg);
} /* 模拟服务器通过reply返回数据给rpc client,然后rpc client处理收到的rpc返回值 */
{
rpc_response_str = rpc_server.reply(1, 1, 2); /* (1,1,2)中的1为调用方的req_id, (1,2)为返回值 */
rpc_client.handleResponse(rpc_response_str);
} {
rpc_response_str = rpc_server.reply(2, "hello", "world", 3);
rpc_client.handleResponse(rpc_response_str);
} cin.get();
return 0;
}
RPC"框架"代码地址: https://github.com/IronsDu/accumulation-dev/blob/master/utils/rpc_test.cpp 。
欢迎讨论。
---update:
我们来看看最新战果:
class Player : public dodo::rpc
{
public:
Player()
{
registerHandle("player_attack", &Player::attack);
registerHandle("player_hi", &Player::hi);
} private:
template<typename... Args>
void registerHandle(string name, void (Player::*callback)(Args...))
{
def(name.c_str(), [this, callback](Args... args){
(this->*callback)(args...);
});
} private:
void attack(string target)
{
cout << "attack:" << target << endl;
} void hi(string i, string j)
{
cout << i << j << endl;
}
}; Player rpc_server; /*rpc服务器*/
Player rpc_client; /*rpc客户端*/ rpc_request_msg = rpc_client.call("player_attack", "Li Lei");
rpc_server.handleRpc(rpc_request_msg);
rpc_request_msg = rpc_client.call("player_hi", "Hello", "World");
rpc_server.handleRpc(rpc_request_msg);
每一个Player就是一个rpc,再结合网络库,就能很轻松的开发业务逻辑。
智能 RPC框架 (C++)的更多相关文章
- rpc框架之 thrift 学习 1 - 安装 及 hello world
thrift是一个facebook开源的高效RPC框架,其主要特点是跨语言及二进制高效传输(当然,除了二进制,也支持json等常用序列化机制),官网地址:http://thrift.apache.or ...
- RPC框架实践之:Apache Thrift
一.概述 RPC(Remote Procedure Call)即 远程过程调用,说的这么抽象,其实简化理解就是一个节点如何请求另一节点所提供的服务.在文章 微服务调用链追踪中心搭建 一文中模拟出来的调 ...
- SpringCloud Alibaba (四):Dubbo RPC框架
Dubbo简介 Apache Dubbo |ˈdʌbəʊ| 是一款高性能.轻量级的开源Java RPC框架,它提供了三大核心能力:面向接口的远程方法调用,智能容错和负载均衡,以及服务自动注册和发现.致 ...
- RPC 框架 Dubbo 从理解到使用(一)
技术架构演变 单一应用架构 通俗地讲,"单体应用(monolith application)"就是将应用程序的所有功能都打包成一个独立的单元.当网站流量很小时,只需一个应用,将所有 ...
- 什么是RPC,RPC好处,常用的RPC框架
RPC简介 RPC(Remote Procedure Call Protocol)远程过程调用协议.一个通俗的描述是:客户端在不知道调用细节的情况下,调用存在于远程计算机上的某个对象,就像调用本地应用 ...
- 【万字长文】Dubbo 入门总结 ,一款高性能的 Java RPC 框架
这篇文章是我学习整理 Dubbo 的一篇文章,首先大部分内容参考了官网 + 某硅谷的视频,内容讲解进行了重新编排,40多张图片,也都是我修改重制的,虽然一万多字,但是其实也可以看出来,更多的内容集中在 ...
- RPC框架实现 - 通信协议篇
RPC(Remote Procedure Call,远程过程调用)框架是分布式服务的基石,实现RPC框架需要考虑方方面面.其对业务隐藏了底层通信过程(TCP/UDP.打包/解包.序列化/反序列化),使 ...
- 微博轻量级RPC框架Motan
Motan 是微博技术团队研发的基于 Java 的轻量级 RPC 框架,已在微博内部大规模应用多年,每天稳定支撑微博上亿次的内部调用.Motan 基于微博的高并发和高负载场景优化,成为一套简单.易用. ...
- 一个轻量级分布式RPC框架--NettyRpc
1.背景 最近在搜索Netty和Zookeeper方面的文章时,看到了这篇文章<轻量级分布式 RPC 框架>,作者用Zookeeper.Netty和Spring写了一个轻量级的分布式RPC ...
随机推荐
- HD1060Leftmost Digit
Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total Submission( ...
- index structure
1. wordlist 0) 0, 1byte 1) token-id(delta), 8byte 2) doclist-offset(delta), 8byte 3) doc_count, 4byt ...
- 从输入一个URL到页面呈现,网络上都发生了什么?
归纳一下其中涉及到前端的一些基础知识,主要包括:http协议.web标准.w3c标准等. 这个问题虽然只有两个2个动作:输入URL和呈现页面,但这背后发生了很多"有趣" ...
- POJ 1160Post Office
POJ 1160 Post Office 我不知道优化,我只知道最暴力的方法,O(V^3),居然100ms不到的过了 设DP[i][j][k]表示考虑前i个小镇,放了j个邮局,最后一个邮局的所在 ...
- C# JackLib系列之如何获取地球上两经纬度坐标点间的距离
获取地球上两经纬度坐标点间的距离,利用[大圆距离公式] A diagram illustrating great-circle distance (drawn in red) between tw ...
- 找不到类型或命名空间 datarowview
在绑定数据时经常会用到这个句程序:<%# DataBinder.Eval(Container.DataItem,"xxxx")%>或者<%# DataBinder ...
- java去掉jsp标签内容的方法
//去掉内容的标签 public static String removeTag(String count){ try { int tagCheck=-1; do { i ...
- [Netbeans]为面板设置背景图片
与AndroidStudio等类似IDE不同,在Netbeans开发桌面程序时,不可以直接通过src=@drawable 等方法为窗口设置背景图片.这里介绍一种简便的方法: 1:首先,拖动一个面板到f ...
- UI:MVC设计模式
不是因为有些事情难以做到,我们才失去自信:而是因为我们失去了自信,有些事情才显得难以做到.自信的第一步就是去尝试.不是因为有希望才坚持,而是因为坚持才有了希望.坚持尝试,就有可能成功.加油! Xcod ...
- ClassRequestHandler or VendorRequestHandler wIndex must be less than NumIFs
P1_ro:20000EEA ClassRequestHandler ; CODE XREF: USB__HandleSetup+38j P1_ro:20000EEA LDRB R0, [R4,#4] ...