当前位置: 首页 > news >正文

施磊老师rpc(三)

文章目录

  • mprpc框架项目动态库编译
    • 框架生成动态库
    • 框架初始化函数-文件读取
      • 1. 为什么要传入 `argc, argv`
      • 2. 读取参数逻辑
      • 3. 配置文件设计
    • init部分实现
  • mprpc配置文件加载(一)
    • 配置文件加载类
      • 成员变量
      • 主要方法
      • **src/include/mprpcconfig.h**
    • 配置文件
      • **bin/test.conf**
    • 实现配置文件加载类
      • **src/include/mprpcapplication.h**
      • 注意substr第二个参数的自动截断
      • **src/mprpcconfig.cc**
    • 补充测试框架类init
    • 问题1
      • 问题描述:新增源文件后,没重新生成 Makefile,编译器“看不到”新文件
    • 问题2:
  • mprpc配置文件加载(二)
    • cmake添加
      • `-g` 是什么?
      • `Debug` 模式在 CMake 中
    • gdb调试exe的某个源文件
    • 优化读取
    • 测试
    • 错误

mprpc框架项目动态库编译

框架生成动态库

src/CMakeLists.txt

aux_source_directory(. SRC_LIST)
add_library(mprpc SHARED ${SRC_LIST})  # 生成动态库

example/callee/CMakeLists.txt

set(SRC_LIST userservice.cc ../user.pb.cc)add_executable(provider ${SRC_LIST})target_link_libraries(provider mprpc protobuf)  # 链接动态库

正常编译即正确

框架初始化函数-文件读取

MprpcApplication::Init()

1. 为什么要传入 argc, argv

  • 为了支持从命令行读取配置文件路径(如 ./provider -i config.conf
  • 这样可以让多个部署节点通过不同的配置文件指定自己的网络参数、ZooKeeper地址等

2. 读取参数逻辑

  • 使用 getopt(需引入 <unistd.h>
  • 支持参数格式:-i config.conf
  • 如果参数错误或缺失,调用 ShowArgsHelp() 提示用户正确格式,并 exit(EXIT_FAILURE)

3. 配置文件设计

格式为 key=value 的文本文件,包含 4 项内容,例如:

rpcserverip=127.0.0.1
rpcserverport=8000
zookeeperip=127.0.0.1
zookeeperport=2181

init部分实现

getopt函数 自行查看 man 3

RETURN VALUEIf an option was successfully found, then getopt() returns the option character.  If all command-line  optionshave  been  parsed, then getopt() returns -1.  If getopt() encounters an option character that was not in opt‐string, then '?' is returned.  If getopt() encounters an option with a missing argument, then the return valuedepends on the first character in optstring: if it is ':', then ':' is returned; otherwise '?' is returned.
void MprpcApplication::Init(int argc, char **argv)
{if (argc < 3){// std::cout << "error: argc < 3" << std::endl;// exit(1);ShowArgsHelp();exit(EXIT_FAILURE); // EXIT_FAILURE 是一个宏,表示程序异常退出}std::string config_file; // 配置文件// 解析命令行参数int c = 0;while ((c = getopt(argc, argv, "i:")) != -1) // getopt函数解析命令行参数{switch (c){case 'i':config_file = optarg; // optarg是一个全局变量,存储当前选项的参数break;case '?':  // '?'表示没有找到对应的选项std::cout << "invalid option: " << (char)c<< std::endl;ShowArgsHelp();exit(EXIT_FAILURE);case ':':  // ':'表示选项缺少参数std::cout << "need config_file " << std::endl;ShowArgsHelp();exit(EXIT_FAILURE);default:ShowArgsHelp();exit(EXIT_FAILURE);}} // 读取配置文件----单独写 .h和.cc文件----解耦-且 代码 也不是很少}

编译测试一下,

mprpc配置文件加载(一)

配置文件加载类

成员变量

  • 使用 std::unordered_map<std::string, std::string> 存储配置项的键值对。

主要方法

  • load_config_file(const char* config_file)
    
    • 打开配置文件,逐行读取内容。
    • 处理注释行(以 # 开头)、空行、以及前后多余的空格。
    • 解析合法的键值对(通过 = 分隔)。
    • 将键值对存入 unordered_map
  • load(const std::string& key)
    
    • 根据键查询配置项的值。
    • 如果键不存在,返回空字符串。

src/include/mprpcconfig.h

#pragma once#include <unordered_map>
#include <string>// 框架读取配置文件的类
class MprpcConfig
{public:// 负责解析加载配置文件void LoadConfigFile(const char* config_file);// 查询配置项std::string Load(const std::string& key);private:std::unordered_map<std::string, std::string> m_configMap; // 存储配置文件的键值对
};

配置文件

bin/test.conf

# rpc节点的ip地址
rpcserverip=127.0.0.1
# rpc节点的端口
rpcserverport=8000
# zk节点的ip地址
zookeeperip=127.0.0.1
# zk节点的端口
zookeeperport=5000

实现配置文件加载类

src/include/mprpcapplication.h

#include "mprpcconfig.h"....static MprpcConfig m_config; // 配置文件对象

类内的静态成员函数不能直接访问普通的成员变量。

静态成员函数不依赖于类的具体实例,而是属于类本身,因此它不能访问实例成员变量,因为实例成员变量是与具体对象实例相关联的。

多了解字符串类型, 面试问的很多, 要熟悉其各种方法,多运用
见知识补充

注意substr第二个参数的自动截断

src/mprpcconfig.cc

#include "mprpcconfig.h"
#include <iostream>// 负责解析加载配置文件
void MprpcConfig::LoadConfigFile(const char* config_file)  
{FILE* pf = fopen(config_file, "r");if(pf == nullptr){std::cout << "error: config file is not exist" << std::endl;exit(EXIT_FAILURE);}while(!feof(pf)) // feof函数判断文件是否到达末尾{char buffer[512] = {0}; // 定义一个字符数组, 用于存储一行数据fgets(buffer, sizeof(buffer), pf); // 读取一行数据// 去掉多余空格std::string src_buf(buffer); // 将字符数组转换为字符串int idx = src_buf.find_first_not_of(" \t"); // 查找第一个不是空格或制表符的位置if(idx != std::string::npos){src_buf = src_buf.substr(idx, src_buf.size() - idx); // 截取字符串}// 去掉注释if(src_buf[0] == '#' || src_buf.empty()) // 如果是注释或空行{continue; // 跳过 这一行}// 解析配置项idx = src_buf.find('='); // 查找第一个等号的位置if(idx!= std::string::npos){std::string key = src_buf.substr(0, idx); // 截取键std::string value = src_buf.substr(idx + 1, src_buf.size() - idx - 1); // 截取值// 可以考虑conf书写不规范, 去掉多余空格// 存储配置项m_configMap.insert({key, value}); // 将键值对插入到map中}}}// 查询配置项
std::string MprpcConfig::Load(const std::string& key)
{// return m_configMap[key]; // 错误的, 不要用中括号, 不存在 会自动插入一个空值auto it = m_configMap.find(key); // 查找键if(it == m_configMap.end()) // 如果没有找到{std::cout << "error: key is not exist" << std::endl;return "";}return it->second; // 返回值
}

补充测试框架类init

MprpcConfig MprpcApplication::m_config;  // 静态成员变量, 需要在类外初始化void MprpcApplication::Init(int argc, char **argv)
{if (argc < 3){// std::cout << "error: argc < 3" << std::endl;// exit(1);ShowArgsHelp();exit(EXIT_FAILURE); // EXIT_FAILURE 是一个宏,表示程序异常退出}std::string config_file; // 配置文件// 解析命令行参数int c = 0;while ((c = getopt(argc, argv, "i:")) != -1) // getopt函数解析命令行参数{switch (c){case 'i':config_file = optarg; // optarg是一个全局变量,存储当前选项的参数break;case '?':  // '?'表示没有找到对应的选项ShowArgsHelp();exit(EXIT_FAILURE);case ':':  // ':'表示选项缺少参数ShowArgsHelp();exit(EXIT_FAILURE);default:ShowArgsHelp();exit(EXIT_FAILURE);}m_config.LoadConfigFile(config_file.c_str()); // 加载配置文件, config_file是一个std::string类型的变量, 文件名字std::cout<<"rpcserverip:"<<m_config.Load("rpcserverip")<<std::endl;std::cout<<"rpcserverport:"<<m_config.Load("rpcserverport")<<std::endl;std::cout<<"zookeeperip:"<<m_config.Load("zookeeperip")<<std::endl;std::cout<<"zookeeperport:"<<m_config.Load("zookeeperport")<<std::endl;} // 读取配置文件----单独写 .h和.cc文件----解耦-且 代码 也不是很少// rpcserver_ip   rpcserver_port  zookeeper_ip zookeeper_port }

问题1

直接进行cmake 编译, 会报错!!

[build] /usr/bin/ld: ../../src/libmprpc.so: undefined reference to `MprpcApplication::m_config'
[build] collect2: error: ld returned 1 exit status
[build] gmake[2]: *** [example/callee/CMakeFiles/provider.dir/build.make:114: ../bin/provider] Error 1
[build] gmake[1]: *** [CMakeFiles/Makefile2:157: example/callee/CMakeFiles/provider.dir/all] Error 2
[build] gmake: *** [Makefile:91: all] Error 2

问题描述:新增源文件后,没重新生成 Makefile,编译器“看不到”新文件

场景如下:

  • CMake 使用了 file(GLOB ...)aux_source_directory(...) 收集源文件;
  • 然后你手动在代码目录中添加了新的 .cpp 文件;
  • 但是你没有清除 CMake 缓存或重新运行 cmake 命令
  • 结果:Makefile 没更新,新文件不会被编译,也不在目标构建中。

问题2:

[!WARNING]

MprpcConfig MprpcApplication::m_config; // 静态成员变量, 需要在类外初始化

mprpc配置文件加载(二)

cmake添加

-g 是什么?

  • -g 是 GCC/Clang 的编译器选项,用于生成调试信息(供 GDB 等调试器使用)。
  • 编译出来的程序体积更大,但可以逐行调试、查看变量值等。

Debug 模式在 CMake 中

cmake复制编辑
set(CMAKE_BUILD_TYPE "Debug")

这条语句告诉 CMake 使用 Debug 配置,其效果通常是:

  • 自动添加 -g
  • 开启 -O0(不优化,便于调试)
  • 设置调试宏(如 _DEBUG
set(CMAKE_BUILD_TYPE "Debug")  # 设置 Debug 模式并开启调试信息

gdb调试exe的某个源文件

gdb ./provider
break mprpcconfig.cc:<行数>

优化读取

为了 适应 更多不规范的 conf 文件

封装一下 去除前后空格—> 不仅可以一行前后去空格, 还可以取出 = 前后 再去空格

void MprpcConfig::Trim(std::string &src_buf);
// 去除前后空格
void MprpcConfig::Trim(std::string &src_buf)
{int idx = src_buf.find_first_not_of(" "); // 查找第一个不是空格的位置if (idx != std::string::npos){src_buf = src_buf.substr(idx, src_buf.size() - idx); // 截取字符串}idx = src_buf.find_last_not_of(" "); // 查找最后一个不是空格的位置if (idx != std::string::npos){src_buf = src_buf.substr(0, idx + 1); // 截取字符串}
}

src/mprpcconfig.cc

// 负责解析加载配置文件
void MprpcConfig::LoadConfigFile(const char *config_file)
{FILE *pf = fopen(config_file, "r");if (pf == nullptr){std::cout << "error: config file is not exist" << std::endl;exit(EXIT_FAILURE);}while (!feof(pf)) // feof函数判断文件是否到达末尾{char buffer[512] = {0};            // 定义一个字符数组, 用于存储一行数据fgets(buffer, sizeof(buffer), pf); // 读取一行数据// 去掉多余空格std::string read_buf(buffer); // 将字符数组转换为字符串Trim(read_buf);              // 去掉前后空格// 去掉注释if (read_buf[0] == '#' || read_buf.empty()) // 如果是注释或空行{continue; // 跳过 这一行}// 解析配置项int idx = read_buf.find('='); // 查找第一个等号的位置if (idx != std::string::npos){std::string key = read_buf.substr(0, idx);                              // 截取键Trim(key); // 去掉前后空格// 先去\nint endidx = read_buf.find_last_not_of("\r\n", read_buf.size()-1); // 查找最后一个不是回车或换行的位置std::string value;if (endidx != std::string::npos){value = read_buf.substr(idx+1, endidx-idx); // 截取字符串}Trim(value); // 再去掉前后空格// 下面这段不对, \n本身就是最后一个, 要是 \n之前紧挨着空格呢/*// 还有换行idx = value.find_last_not_of("\r\n"); // 查找最后一个不是回车或换行的位置if (idx != std::string::npos){value = value.substr(0, idx); // 截取值}*/// 存储配置项m_configMap.insert({key, value}); // 将键值对插入到map中}}
}

测试

自行测试

可能还不是最好的 处理 所有情况, 但是 要给用户 一定容错!!

错误

出现错误, 不要着急解决, 先去定位!!!

http://www.xdnf.cn/news/266527.html

相关文章:

  • Docker安装Ollama及使用Ollama部署大模型
  • 二极管反向恢复的定义和原理
  • SQL语句--postgis语句(矢量数据的定义与操作)
  • REINFORCE蒙特卡罗策略梯度算法详解:python从零实现
  • STM32 DMA直接存储器存取
  • 解码响应式 Web 设计:原理、技术与优劣势全解析
  • C++代码随想录刷题知识分享-----142.环形链表II
  • 希洛激活器策略思路
  • n8n工作流自动化平台的实操:Cannot find module ‘iconv-lite‘
  • 生成式 AI 与 AI 的区别
  • DeepSeek实战--LLM微调
  • LeetCode算法题 (设计链表)Day16!!!C/C++
  • 「Mac畅玩AIGC与多模态16」开发篇12 - 多节点串联与输出合并的工作流示例
  • ipvsadm,是一个什么工具?
  • 中国 AIGC 确权革命:“AI 创意・中国” 平台上线,存证成本降至 0.1 元 / 件
  • CAN网桥中继隔离抗干扰集线器重映射一进一出CAN扩展CAN Bridge
  • 在Java项目中实现本地语音识别与热点检测,并集成阿里云智能语音服务
  • Dubbo(92)如何在微服务架构中应用Dubbo?
  • 深入理解C++类型转换:从基础到高级应用
  • 糖尿病筛查常识---秋浦四郎
  • 计网_可靠传输ARQ机制
  • neo4j初尝试
  • Java从入门到精通 - Java语法
  • C++ 简单工厂模式详解
  • QT6 源(72):阅读与注释单选框这个类型的按钮 QRadioButton,及各种属性验证,
  • 【Linux知识】find命令行使用详解
  • 数据结构*队列
  • nessus最新版本安装教程+windows一键更新最新插件
  • 计算机网络-同等学力计算机综合真题及答案
  • 【AI零件】openrouter.ai生成密钥的操作