C++中protobuf 动态加载.proto文件

C++中protobuf 动态加载.proto文件

环境:

复制代码
protobuf: v27.3(2024-08-01)
abseil: 20240722.0

文章目录

  • [C++中protobuf 动态加载.proto文件](#C++中protobuf 动态加载.proto文件)
    • 前言
    • [1. 通讯录 addressbook.proto](#1. 通讯录 addressbook.proto)
    • [2. C++中测试动态加载.proto文件并输出json](#2. C++中测试动态加载.proto文件并输出json)
    • [3. 结果](#3. 结果)

前言

protobuf动态加载.proto文件,可以不生成cpp文件的情况下操作Message对象。

动态加载方式的性能会稍差一些。

1. 通讯录 addressbook.proto

addressbook.proto

复制代码
syntax = "proto3";

option optimize_for = LITE_RUNTIME;  // MessageLite

package com.test;

message Person {
  string name = 1;
  int32 age = 2;
  string phone = 3;
}

message AddressBook {
  repeated Person people = 1;
}

2. C++中测试动态加载.proto文件并输出json

main.cpp

复制代码
#include <iostream>

#include <google/protobuf/compiler/importer.h>
#include <google/protobuf/dynamic_message.h>

#include <google/protobuf/util/json_util.h> // MessageToJsonString JsonStringToMessage


int main(int argc, char *argv[])
{
    // 1. Setup Importer to dynamically load .proto files
    google::protobuf::compiler::DiskSourceTree sourceTree;
    sourceTree.MapPath("", ".");
    google::protobuf::compiler::MultiFileErrorCollector* errorCollector = nullptr;
    google::protobuf::compiler::Importer importer(&sourceTree, errorCollector);

    // 2. Import the .proto file
    const google::protobuf::FileDescriptor* addressBookFileDescriptor = importer.Import("addressbook.proto");
    if (!addressBookFileDescriptor) 
    {
    std::cerr << "Failed to import .proto file." << std::endl;
    return -1;
    }

    // 3. Find the message descriptor
    const google::protobuf::Descriptor* addressBookMessageDescriptor = addressBookFileDescriptor->FindMessageTypeByName("AddressBook");
    if (!addressBookMessageDescriptor) {
    std::cerr << "Message type 'AddressBook' not found in .proto file." << std::endl;
    return -1;
    }

    // 4. Create a dynamic message
    google::protobuf::DynamicMessageFactory factory;
    const google::protobuf::Message* prototype = factory.GetPrototype(addressBookMessageDescriptor);
    google::protobuf::Message* message = prototype->New();

    // 5. JSON to PB Message
    std::string jsonStr = R"({"people":[{"name":"xiaoming","age":30,"phone":"13012345678"}]})";
    google::protobuf::json::JsonStringToMessage(jsonStr, message);
    
    // 6. printf PB JSON
    std::string result;
    google::protobuf::json::MessageToJsonString(*message, &result);
    std::cout << "Dynamic Message From JSON - " << result << std::endl;

    // 7. clear
    delete message;

    getchar();
    return 0;
}

CMakeLists.txt

动态加载依赖libprotoc

复制代码
cmake_minimum_required(VERSION 3.10)

project(main)

# protobuf
add_definitions(-DPROTOBUF_USE_DLLS)
include_directories(include)
link_directories(lib)

add_executable(${PROJECT_NAME} main.cpp)

target_link_libraries(${PROJECT_NAME} libprotoc libprotobuf abseil_dll)

目录结构

复制代码
tree
.
+--- addressbook.proto
+--- CMakeLists.txt
+--- include
+--- lib
+--- main.cpp

3. 结果

复制代码
Dynamic Message From JSON - {"people":[{"name":"xiaoming","age":30,"phone":"13012345678"}]}
相关推荐
2401_831920749 分钟前
分布式系统安全通信
开发语言·c++·算法
2401_8772742440 分钟前
从匿名管道到 Master-Slave 进程池:Linux 进程间通信深度实践
linux·服务器·c++
汉克老师1 小时前
GESP5级C++考试语法知识(八、链表(三)循环链表)
c++·约瑟夫问题·循环链表·gesp5级·gesp五级
阿贵---1 小时前
C++中的RAII技术深入
开发语言·c++·算法
PiKaMouse.2 小时前
navigation2-humble从零带读笔记第一篇:nav2_core
c++·算法·机器人
lightqjx2 小时前
【算法】二分算法
c++·算法·leetcode·二分算法·二分模板
Irissgwe3 小时前
进程间通信
linux·服务器·网络·c++·进程间通信
add45a3 小时前
C++编译期数据结构
开发语言·c++·算法
灰色小旋风3 小时前
力扣21 合并两个有序链表(C++)
c++·leetcode·链表
Laurence4 小时前
Qt 前后端通信(QWebChannel Js / C++ 互操作):原理、示例、步骤解说
前端·javascript·c++·后端·交互·qwebchannel·互操作