0

0

c++怎么实现一个简单的RPC框架_c++远程过程调用机制与网络通信实现

冰火之心

冰火之心

发布时间:2025-11-17 16:14:02

|

997人浏览过

|

来源于php中文网

原创

答案:该C++ RPC框架通过函数注册、JSON序列化和TCP通信实现远程调用。客户端调用本地桩函数,参数与函数名被序列化后发送至服务端;服务端反序列化请求,查找注册函数并执行,将结果序列化回传;核心包括RpcRequest/RpcResponse结构定义、RpcServer处理客户端连接与函数执行、RpcClient封装请求发送与响应接收;示例中add函数注册与调用验证了整数相加功能,展示了RPC基本流程。

c++怎么实现一个简单的rpc框架_c++远程过程调用机制与网络通信实现

实现一个简单的 C++ RPC(Remote Procedure Call)框架,核心是让客户端像调用本地函数一样调用远程服务器上的函数。这需要解决函数名映射、参数序列化、网络传输和响应返回等问题。下面从基本原理出发,逐步构建一个轻量级的 RPC 框架。

1. RPC 基本流程

一个最简 RPC 调用过程包括:

  • 客户端调用本地“桩”函数(stub)
  • 桩函数将函数名和参数序列化
  • 通过网络发送到服务端
  • 服务端反序列化,查找并执行对应函数
  • 将结果序列化后返回客户端
  • 客户端反序列化结果并返回

2. 序列化与函数注册

使用 JSON 或自定义格式进行参数序列化。这里以 JSON 为例(可用 nlohmann/json 简化操作)。

示例:定义请求结构

struct RpcRequest {
    std::string func_name;
    nlohmann::json args;
};

struct RpcResponse {
    int code = 0;
    std::string msg;
    nlohmann::json result;
};

服务端需维护函数注册表

立即学习C++免费学习笔记(深入)”;

class RpcServer {
public:
    using Func = std::function;

    void register_function(const std::string& name, Func f) {
        functions[name] = f;
    }

private:
    std::unordered_map functions;
};

3. 网络通信设计(基于 TCP)

使用 socket 实现简单 TCP 通信。客户端发送序列化后的请求,服务端接收并处理。

服务端监听循环示例:

腾讯AI 开放平台
腾讯AI 开放平台

腾讯AI开放平台

下载

void RpcServer::run(int port) {
    int server_fd = socket(AF_INET, SOCK_STREAM, 0);
    struct sockaddr_in addr;
    addr.sin_family = AF_INET;
    addr.sin_addr.s_addr = INADDR_ANY;
    addr.sin_port = htons(port);

    bind(server_fd, (struct sockaddr*)&addr, sizeof(addr));
    listen(server_fd, 5);

    while (true) {
        int client_fd = accept(server_fd, nullptr, nullptr);
        handle_client(client_fd);
    }
}

处理客户端请求:

void RpcServer::handle_client(int client_fd) {
    char buffer[1024] = {0};
    read(client_fd, buffer, 1024);

    // 解析请求
    auto req_json = nlohmann::json::parse(buffer);
    RpcRequest req{
        req_json["func_name"],
        req_json["args"]
    };

    // 查找并执行函数
    auto it = functions.find(req.func_name);
    RpcResponse resp;
    if (it != functions.end()) {
        try {
            resp.result = it->second(req.args);
        } catch (...) {
            resp.code = -1;
            resp.msg = "error executing function";
        }
    } else {
        resp.code = -1;
        resp.msg = "function not found";
    }

    // 返回结果
    std::string resp_str = resp.result.dump();
    send(client_fd, resp_str.c_str(), resp_str.size(), 0);
    close(client_fd);
}

4. 客户端调用封装

客户端封装发送请求和接收结果的过程:

class RpcClient {
public:
    RpcClient(const std::string& ip, int port) {
        sockfd = socket(AF_INET, SOCK_STREAM, 0);
        struct sockaddr_in serv_addr;
        serv_addr.sin_family = AF_INET;
        serv_addr.sin_port = htons(port);
        inet_pton(AF_INET, ip.c_str(), &serv_addr.sin_addr);
        connect(sockfd, (struct sockaddr*)&serv_addr, sizeof(serv_addr));
    }

    std::string call(const std::string& func, const nlohmann::json& args) {
        RpcRequest req{func, args};
        std::string data = nlohmann::json(req).dump();

        send(sockfd, data.c_str(), data.size(), 0);

        char buffer[1024] = {0};
        int n = read(sockfd, buffer, 1024);
        return std::string(buffer, n);
    }

private:
    int sockfd;
};

这样,用户可以通过 client.call("add", {{"a", 1}, {"b", 2}}) 远程调用服务端函数。

5. 示例:注册一个加法函数

在服务端注册函数:

server.register_function("add", [](const nlohmann::json& args) {
    int a = args.value("a", 0);
    int b = args.value("b", 0);
    return nlohmann::json(a + b).dump();
});

客户端调用:

RpcClient client("127.0.0.1", 8080);
std::string result = client.call("add", {{"a", 3}, {"b", 4}});
int sum = nlohmann::json::parse(result);
// sum == 7

基本上就这些。这个简易框架展示了 RPC 的核心机制:函数注册、序列化、网络通信和调用转发。实际项目中可扩展支持更多数据类型、错误处理、超时机制和多线程。不复杂但容易忽略细节,比如连接管理或异常安全。

相关专题

更多
json数据格式
json数据格式

JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

409

2023.08.07

json是什么
json是什么

JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

532

2023.08.23

jquery怎么操作json
jquery怎么操作json

操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

309

2023.10.13

go语言处理json数据方法
go语言处理json数据方法

本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

74

2025.09.10

数据类型有哪几种
数据类型有哪几种

数据类型有整型、浮点型、字符型、字符串型、布尔型、数组、结构体和枚举等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

299

2023.10.31

php数据类型
php数据类型

本专题整合了php数据类型相关内容,阅读专题下面的文章了解更多详细内容。

222

2025.10.31

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

480

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

143

2025.12.24

Java 项目构建与依赖管理(Maven / Gradle)
Java 项目构建与依赖管理(Maven / Gradle)

本专题系统讲解 Java 项目构建与依赖管理的完整体系,重点覆盖 Maven 与 Gradle 的核心概念、项目生命周期、依赖冲突解决、多模块项目管理、构建加速与版本发布规范。通过真实项目结构示例,帮助学习者掌握 从零搭建、维护到发布 Java 工程的标准化流程,提升在实际团队开发中的工程能力与协作效率。

10

2026.01.12

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 8.2万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号