Windows平台下CMake工程中使用protobuf

Windows平台下的CMake工程中使用protobuf

本文档介绍如何在Windows平台下的CMake工程中正确配置和使用Protobuf库,解决链接错误问题。

本文在以下环境中使用静态链接库测试通过:

环境配置

  • Windows 10
  • Visual Studio 2026
  • CMake 4.1.1-msvc1
  • Ninja 1.12.1
  • Clang 21.1.5
  • Protobuf 33.2

注:以上的CMake版本是Visual Studio 2026自带的版本。在环境变量添加以下配置即可: <VisualStudio 2026 安装目录>/Common7/IDE/CommonExtensions/Microsoft/CMake/CMake/bin

操作步骤

一、初始化Protobuf配置

本文使用源码编译配置Protobuf库。

  1. 下载Protobuf源码Protobuf 33.2 选择下载 protobuf-33.2.zip
  2. 解压 protobuf-33.2.zip
  3. 在解压后的protobuf-33.2目录下,创建 build.bat 写入以下内容,然后执行这个脚本。
bat 复制代码
@echo off
cmake ./ ^
  -B build ^
  -G "Ninja" ^
  -D CMAKE_C_COMPILER=clang ^
  -D CMAKE_CXX_COMPILER=clang++ ^
  -D protobuf_BUILD_TESTS=OFF ^
  -D CMAKE_BUILD_TYPE=Release ^
  -D CMAKE_INSTALL_PREFIX=./install ^
  -D CMAKE_CXX_STANDARD=17 ^
  -D CMAKE_CXX_STANDARD_REQUIRED=ON

cmake --build build --config Release

cmake --install build
  1. 编译完成之后,会在 protobuf-33.2/install 目录下,会生成 binincludelib 三个目录。里面包括Protobuf的头文件库文件protoc 编译工具等。

  2. protobuf-33.2/install/bin 目录添加到环境变量中,即可全局使用 protoc 编译工具。

二、在CMake工程中使用Protobuf库

  1. 注意设置工程CMAKE_BUILD_TYPERelease, 保持和Protobuf库的编译类型一致。

  2. 在CMakeLists.txt 头部添加 cmake_policy(SET CMP0091 NEW) 来设置新的CMake策略。

cmake 复制代码
# 在 project() 之前添加;主要是为了能成功设置 CMAKE_MSVC_RUNTIME_LIBRARY
cmake_policy(SET CMP0091 NEW)

# 在 project() 之后设置 CMAKE_MSVC_RUNTIME_LIBRARY 解决 RuntimeLibrary 链接错误
if (CMAKE_BUILD_TYPE STREQUAL "Debug")
  set(CMAKE_MSVC_RUNTIME_LIBRARY "MultiThreaded$<$<CONFIG:Debug>:Debug>")
else()
  set(CMAKE_MSVC_RUNTIME_LIBRARY "MultiThreaded")
endif()

# 设置 C++ 标准为 C++17 因为 Protobuf 依赖 Absl 库,而 Absl 库要求 C++17 标准
set(CMAKE_CXX_STANDARD 17)
set(CMAKE_CXX_STANDARD_REQUIRED ON)

# 添加 protobuf 相关配置
# 设置 Protobuf 库的搜索路径 请替换为你自己的完整安装路径
set(CMAKE_PREFIX_PATH "D:/protobuf-33.2/install")

# 查找Protobuf及Protobuf所依赖的包
find_package(Protobuf REQUIRED)
# 以下两个都是Protobuf所依赖的库
find_package(Absl REQUIRED)
find_package(utf8_range REQUIRED)


# 链接utf8_range库 解决Protobuf链接错误
target_link_libraries(${PROJECT_NAME} utf8_range::utf8_range)

# 链接Absl库 解决Protobuf链接错误
target_link_libraries(${PROJECT_NAME} absl::check absl::log_flags absl::statusor)

# 链接Protobuf库
target_link_libraries(${PROJECT_NAME} ${Protobuf_LIBRARIES})

注: 直接链接Protobuf库, 会有链接错误,必须要链接utf8_rangeAbsl库。

  1. 编译工程,即可成功使用Protobuf库。

三、写在最后

  1. 以上配置仅适用于Windows平台下的CMake工程。
  2. 也可以使用其他的编译器,如MSVC等,需要确保编译Protobuf库和使用Protobuf的工程都使用相同的编译器。参考编译bat脚本
bat 复制代码
cmake ./ ^
  -B build ^
  -G "Visual Studio 18 2026" ^
  -D protobuf_BUILD_TESTS=OFF ^
  -D CMAKE_BUILD_TYPE=Release ^
  -D CMAKE_INSTALL_PREFIX=./install ^
  -D CMAKE_CXX_STANDARD=17 ^
  -D CMAKE_CXX_STANDARD_REQUIRED=ON

cmake --build build --config Release

cmake --install build
  1. 过程中可以遇到以下错误:
  • RuntimeLibrary 不匹配错误:
txt 复制代码
[build] lld-link: error: /failifmismatch: mismatch detected for 'RuntimeLibrary':
[build] >>> CMakeFiles/test_cpp.dir/cpp/test.pb.cc.obj has value MD_DynamicRelease
[build] >>> protobuf.lib(message.cc.obj) has value MT_StaticRelease

主要检查工程cmake_policyCMAKE_MSVC_RUNTIME_LIBRARY是否正确设置;参考文档

其他版本的cmake参考对应版本设置cmake_policy即可。

  • Protobuf链接错误:
txt 复制代码
lld-link: error: undefined symbol: utf8_range_IsValid
[build] >>> referenced by protobuf.lib(generated_message_tctable_lite.cc.obj):(public: static char const * __cdecl google::protobuf::internal::TcParser::FastUS1(class google::protobuf::MessageLite *, char const *, class google::protobuf::internal::ParseContext *, struct google::protobuf::internal::TcFieldData, struct google::protobuf::internal::TcParseTableBase const *, unsigned __int64))

[build] lld-link: error: undefined symbol: public: void __cdecl absl::lts_20250512::status_internal::StatusRep::Unref(void) const
[build] >>> referenced by CMakeFiles/test_cpp.dir/cpp/main.cpp.obj:(main)
[build] >>> referenced by CMakeFiles/test_cpp.dir/cpp/main.cpp.obj:(public: __cdecl absl::lts_20250512::Status::~Status(void))

主要检查工程是否链接了utf8_rangeAbsl的相关库。

相关推荐
-dzk-2 小时前
【代码随想录】LC 59.螺旋矩阵 II
c++·线性代数·算法·矩阵·模拟
m0_706653232 小时前
C++编译期数组操作
开发语言·c++·算法
qq_423233903 小时前
C++与Python混合编程实战
开发语言·c++·算法
m0_715575343 小时前
分布式任务调度系统
开发语言·c++·算法
CSDN_RTKLIB3 小时前
简化版unique_ptr说明其本质
c++
naruto_lnq3 小时前
泛型编程与STL设计思想
开发语言·c++·算法
m0_748708054 小时前
C++中的观察者模式实战
开发语言·c++·算法
时光找茬4 小时前
【瑞萨AI挑战赛-FPB-RA6E2】+ 从零开始:FPB-RA6E2 开箱测评与 e2 studio 环境配置
c++·单片机·边缘计算
qq_537562674 小时前
跨语言调用C++接口
开发语言·c++·算法
猷咪5 小时前
C++基础
开发语言·c++