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,如何安装的就不说了, ...
随机推荐
- Android 的异步消息处理机制
前言 Android中的异步消息处理机制主要有四部分:Message.Handler.MessageQuene.Looper.这一消息处理机制也称为Handler机制.Handler机制是支撑整个An ...
- 不要轻易定义指向std::vector中的元素的指针
类应该是被封装的,类的用户通过接口使用类提供的功能,而不必关心类的内部如何实现.然而,C++标准库容器 std::vector 的实现渗透到了接口中来.对于以下代码: const int pushNu ...
- JAVAweek5
学习内容 进制 1.(十进制):752=2*10(0)+5*10(1)0+7*10(2)=752 (二进制):1011(二进制的数)=1*2(0)+1*2(1)+0*2(2)+1*2(3) = 1 ...
- [Codechef REBXOR]Nikitosh and xor (Trie,异或)
题目传送门 分析:首次考虑暴力枚举 \(l_{1},r_{1},l_{2},r_{2}\),配合前缀和时间复杂度 \(O(N^{4})\),需要想办法优化.对于这种两段区间不重合的,我们考虑枚举两段区 ...
- 叮咚,你的微信年度聊天报告请查收「GitHub 热点速览」
本周热点项目 WeChatMsg 是一个微信记录提取工具,据说它还能帮你分析聊天记录.生成你的年度聊天报告.而又到了年底,部分不幸的小伙伴要开始写年度总结了,这时候 self-operating-co ...
- Aiganize微信小程序开发手册二代
根据此表格, 现有三个模块: 活动模块 聊天模块 影子模块 现活动模块交与:赵坤亮.郝文章做 现聊天模块与用户信息交与:葛方杰.陈金鹏做影子模块待定,现做完那两块,已经有不错的用户体验了.
- 如何将 performance_schema 中的 TIMER 字段转换为日期时间
问题 最近有好几个朋友问,如何将 performance_schema.events_statements_xxx 中的 TIMER 字段(主要是TIMER_START和TIMER_END)转换为日期 ...
- 累加器Adder
① java8引⼊的,相⽐较是⼀个⽐较新的类 ② ⾼并发下LogAdder⽐AtomicLog效率⾼,不过本质是空间换时间 ③ 竞争激烈的时候,LongAdder把不同线程对应到不同的Cell上进⾏修 ...
- C++学习笔记一:windows系统配置C++开发环境(VS code+g++/clang++)
1.下载vscode 官网下载地址: https://code.visualstudio.com/ 安装时选择把软件加入到环境变量中这个选项 2.打开vscode,安装c/c++扩展插件 3.下载gc ...
- 『Flutter』开发环境搭建
1.前言 大家好,我是 BNTang,今天给大家介绍一下 Flutter 的开发环境搭建.在之前我已经将 Dart 的基本语法给大家介绍了,所以今天就不再介绍 Dart 的基本语法了,直接进入 Flu ...