CMake + Protobuf 自动生成 cpp 文件(pb.h, pb.cc)
本文介绍在 macOS 系统下 cmake 和 protobuf 一起使用的一种方式——使用 cmake 自动编译 proto 文件为 pb.cc 和 pb.h 文件。
protobuf
protobuf 是谷歌发布的一种数据封装协议, 用于数据传输。在使用时需要定义好 proto 文件, 然后用 protoc 工具编译为 pb.h 和 pb.cc 文件。打开命令行输入如下命令
protoc -I=${proto_file_dir} --cpp_out=${pb_file_dir} *.proto
这里面有三个参数, -I 表示 proto 文件的路径; --cpp_out 表示输出路径; 最后一个参数表示需要被编译为 pb.h 和 pb.cc 文件的 proto 文件, *.proto 表示 -I 路径下的所有 proto 文件。
好了, 正文开始。
背景介绍
文件结构
.
├── CMakeLists.txt
├── originFile
│ └── proto1
│ ├── proto1.proto
│ ├── proto2.proto
│ └── proto3.proto
│ └── proto2
│ ├── proto4.proto
│ ├── proto5.proto
│ └── proto6.proto
目标
我们要把 originFile 中的文件编译为 pb 文件。
根据上面的指令, 我们可以很方便很快捷的生成 pb 文件,似乎没有什么不妥。但是在不同的主机上使用不同版本的 protobuf 时, 都需要重新使用指令来生成 pb 文件(因为某一版本的 protobuf 生成的 pb 文件, 不能被其他版本的 protobuf 使用)。为了简化这一过程, 我们把生成 pb 文件的事,交给 cmake 来做, 在 cmake 过程中自动去根据 proto 文件生成 pb 文件。
方法
不指定输出路径
直接上 CmakeLists.txt 文件的内容
# 项目基本配置
cmake_minimum_required(VERSION 3.5)
project(common_pb)
# 获取protobuf库
find_package(Protobuf REQUIRED)
#设置输出路径
SET(PROTO_DIR ${CMAKE_SOURCE_DIR}/proto/originFile)
#获取需要编译的proto文件
file(GLOB_RECURSE MSG_PROTOS_1 ${PROTO_DIR}/proto1/*.proto)
file(GLOB_RECURSE MSG_PROTOS_2 ${PROTO_DIR}/proto2/*.proto)
# 将 proto 编译为 pb
PROTOBUF_GENERATE_CPP(SOURCES_1 HEADERS_1 ${MSG_PROTOS_1})
PROTOBUF_GENERATE_CPP(SOURCES_2 HEADERS_2 ${MSG_PROTOS_2})
这种方式生成的 pb 文件输出在 build 文件夹里面,且不论 proto 文件结构如何, 生成的 pb 文件都会被放在同一个目录下。
这里有另一种方式,来指定输出文件的路径。
指定输出路径
cmake_minimum_required(VERSION 3.5)
project(common_pb)
find_package(Protobuf REQUIRED)
#设置输出路径
SET(PROTO_DIR ${CMAKE_SOURCE_DIR}/originFile/)
SET(PB_DIR ${CMAKE_SOURCE_DIR}/pbCCFile)
message("[PROTO_DIR]" ${PROTO_DIR})
#设置protoc的搜索路径
LIST(APPEND PROTO_FLAGS -I${PROTO_DIR})
#获取需要编译的proto文件
file(GLOB_RECURSE MSG_PROTOS ${PROTO_DIR}/*.proto)
foreach(msg ${MSG_PROTOS})
get_filename_component(FIL_NAME ${msg} NAME_WE)
get_filename_component(FILE_PATH_ABS ${msg} ABSOLUTE)
string(REPLACE "${PROTO_DIR}" "" FILE_PATH_REL ${FILE_PATH_ABS})
string(REPLACE ".proto" ".pb.cc" FILE_NAME_CC ${FILE_PATH_REL})
string(REPLACE ".proto" ".pb.h" FILE_NAME_H ${FILE_PATH_REL})
message("[pb file relative path] " ${FILE_NAME_CC})
# 使用自定义命令
add_custom_command(
OUTPUT
"${PB_DIR}/${FILE_NAME_CC}"
"${PB_DIR}/${FILE_NAME_H}"
COMMAND ${PROTOBUF_PROTOC_EXECUTABLE}
ARGS
--cpp_out ${PB_DIR}
-I ${PROTO_DIR}
${msg}
DEPENDS ${msg}
COMMENT "Running C++ protocol buffer compiler on ${msg}"
VERBATIM
)
set(PROTO_SRC ${PROTO_SRC} ${PB_DIR}/${FILE_NAME_CC})
set(PROTO_INC ${PROTO_INC} ${PB_DIR}/${FILE_NAME_H})
endforeach()
# 设置文件属性为 GENERATED
set_source_files_properties(${PROTO_SRC} ${PROTO_INC} PROPERTIES GENERATED TRUE)
通过这种方式生成的 pb 可以指定输出路径。 这种方式需要自行管理文件依赖和导入路径。
CMake + Protobuf 自动生成 cpp 文件(pb.h, pb.cc)的更多相关文章
- vs2017自动生成的#include“stdafx.h”详解及解决方案
vs2017自动生成的#include“stdafx.h”详解及解决方案 问题描述: 在高版本的Visual Studio的默认设置中,会出现这么一个现象,在新建项目之后,项目会自动生成#includ ...
- linux下使用automake工具自动生成makefile文件
linux环境下,当项目工程很大的时候,编译的过程很复杂,所以需要使用make工具,自动进行编译安装,但是手写makefile文件比较复杂,所幸在GNU的计划中,设计出了一种叫做Autoconf/Au ...
- 使用automake等命令自动生成Makefile文件 (转载)
使用automake等命令自动生成Makefile文件 Linux下编程时,为了方便编译,往往使用Makefile文件自动完成编译,但是Makefile文件本身的书写十分复杂,规则很多.好在Lin ...
- 程序自动生成Dump文件
前言:通过drwtsn32.NTSD.CDB等调试工具生成Dump文件, drwtsn32存在的缺点虽然NTSD.CDB可以完全解决,但并不是所有的操作系统中都安装了NTSD.CDB等调试工具.了解了 ...
- Makefile自动生成头文件依赖
前言 Makefile自动生成头文件依赖是很常用的功能,本文的目的是想尽量详细说明其中的原理和过程. Makefile模板 首先给出一个本人在小项目中常用的Makefile模板,支持自动生成头文件依赖 ...
- 程序自动生成Dump文件()
前言:通过drwtsn32.NTSD.CDB等调试工具生成Dump文件, drwtsn32存在的缺点虽然NTSD.CDB可以完全解决,但并不是所有的操作系统中都安装了NTSD.CDB等调试工具.了解了 ...
- 文件参数化-utp框架之根据yaml文件自动生成python文件+utp运行用例
根据yaml文件自动生成python文件 utp框架: bin目录:存放执行文件(run.py) cases目录:存放生成的用例的python文件(该目录下的文件为根据data目录下的测试用例生成的p ...
- C# protobuf自动更新cs文件
网上的教程大都是手动通过protoc编译, 比较难用 给当前工程添加"Google.Protobuf"和"Grpc.Tools"的引用(通过nuget), 然后 ...
- Makefile中自动生成头文件依赖
为什么需要自动生成头文件依赖? 编译单个源文件时,需要获取文件中包含的头文件的信息,但是一般的Makefile不会在规则中明确写明文件依赖的头文件,所以单独修改头文件后,不会导致包含头文件的源文件重新 ...
- Wix 安装部署(一)同MSBuild 自动生成打包文件
因为项目需要,最近在研究Wix打包部署,园子里也有一些关于wix的博客,方方面面,讲的点各不同.我自己也在测试过程中,写下过程,以供参考.最新版本WiX Toolset v3.7,如何安装的就不说了, ...
随机推荐
- 提高cin cout的速度
ios::sync_with_stdio(false) :用来关闭iostream与stdio的同步,从而提高 cin cout 的效率,但是就不能再用 scanf printf 了,因为不关闭之前 ...
- Selenium4+python被单独定义<div>的动态输入框和二级下拉框要怎么定位?
今天在做练习题的时候,发现几个问题捣鼓了好久,写下这篇来记录 问题一: 有层级的复选框无法定位到二级目录 对于这种拥有二级框的选项无法定位,也不是<select>属性. 我们查看下HTML ...
- 玩转开源 | 搭建 Hugo 管理 Markdown 文档
在工作.学习中,不可避免会要写一些文档:又或者想搭建个简单网站,记录和分享您的生活经验或知识:撰写这些文档中使用 markdown 是一个非常不错的选择,让我们更加聚焦在文档表达的内容上.实际上笔者的 ...
- GMAC网卡Fixed-Link模式
GMAC网卡Fixed-Link模式 GMAC fixed-link固定链接模式,mac与对端的连接方式是写死的,通常用于mac to mac(不排除mac to phy的情况).内核要支持fixed ...
- MarkdownPad 文件访问权限受限导致软件打开后不久闪退解决方法
近几天发现MarkdownPad有一些小问题,打开时会弹出以下报错信息,告诉你打开文件的权限不够 解决方法如下: 1.复制报错信息中的文件路径"Access to the path 'C:\ ...
- Redis本地安装以及使用(详细教程)
Redis 安装 Windows 下载安装 Redis默认端口:6379 整个过程如下: 1.下载连接 https://github.com/tporadowski/redis/releases Re ...
- [USACO2007NOVS] Cow Hurdles S
题目描述 Farmer John wants the cows to prepare for the county jumping competition, so Bessie and the gan ...
- [ABC237G] Range Sort Query
Problem Statement Given is a permutation $P=(P_1,P_2,\ldots,P_N)$ of $1,2,\ldots,N$, and an integer ...
- 2023-12-23:用go语言,一支n个士兵的军队正在趁夜色逃亡,途中遇到一条湍急的大河 敌军在T的时长后到达河面,没到过对岸的士兵都会被消灭 现在军队只找到了1只小船,这船最多能同时坐上2个士兵。
2023-12-23:用go语言,一支n个士兵的军队正在趁夜色逃亡,途中遇到一条湍急的大河 敌军在T的时长后到达河面,没到过对岸的士兵都会被消灭 现在军队只找到了1只小船,这船最多能同时坐上2个士兵. ...
- Odoo16—国际化翻译
开发odoo系统模块的时候,如果一开始就有国际化的需求,无论是模型的定义还是视图的构建,建议使用英语作为第一语言:一方面,英语本身就是一种国际化的语言:另一方面,odoo内置模型字段描述如Create ...