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

CMake指令:option()

目录

1.简介

2.常用场景

3.命令行修改选项

4.与 set(... CACHE BOOL ...) 的区别

5.示例代码

6.总结


1.简介

        在 CMake 中,option() 是一个专门用于定义布尔类型选项的命令,主要用于控制项目的编译行为,比如是否构建测试、是否启用某个功能模块等。它的作用是创建一个可在 CMake 配置阶段进行调整的开关,用户可以通过命令行参数、CMake GUI 或者 ccmake 来修改这些选项的值。

        基本语法:

option(<variable> "<help_text>" [initial value])
  • <variable>:选项的名称,定义后可在 CMake 脚本中通过 ${variable} 来引用。
  • <help_text>:对选项的描述信息,会在 CMake GUI 或者 ccmake 中显示,用于帮助用户理解该选项的用途。
  • [initial value]:选项的初始值,可选参数,默认值为 OFF(即 FALSE), ON(即TRUE)。

        在定义语句执行后才有效,在定义之前均视为未定义(除过在CMake命令中通过-D预定义)

        已定义option选项会存储在CMakeCache.txt中。等价于缓存变量作用。

        如果出现同名的变量(普通或缓存变量),则option会被忽略,或者CMake命令中通过-D预设同名变量,则option也会被忽略。

2.常用场景

1)控制功能模块的构建

option(BUILD_SHARED_LIBS "Build shared libraries instead of static ones" ON)
option(ENABLE_LOGGING "Enable logging feature" OFF)if(ENABLE_LOGGING)target_compile_definitions(myapp PRIVATE ENABLE_LOGGING)
endif()

2)选择性构建测试

option(BUILD_TESTS "Build unit tests" ON)
option(BUILD_EXAMPLES "Build example applications" ON)if(BUILD_TESTS)enable_testing()add_subdirectory(tests)
endif()if(BUILD_EXAMPLES)add_subdirectory(examples)
endif()

3)启用特定平台的功能

option(USE_SYSTEM_LIB "Use system libraries instead of bundled ones" OFF)if(USE_SYSTEM_LIB)find_package(SomeLibrary REQUIRED)target_link_libraries(myapp PRIVATE SomeLibrary::SomeLibrary)
else()add_subdirectory(3rdparty/somelib)target_link_libraries(myapp PRIVATE somelib)
endif()

3.命令行修改选项

方法1:在运行 CMake 配置时,可以通过 -D 参数来覆盖选项的默认值:

# 禁用测试构建
cmake -DBUILD_TESTS=OFF ..# 启用共享库构建
cmake -DBUILD_SHARED_LIBS=ON ..

方法2:通过cache变量方式修改

## 定义option选项
option(${ PROJECT_NAME }_BUILD_EXAMPLE "构建示例模块" OFF)
message(STATUS "KAIZEN_BUILD_EXAMPLE_1: ${KAIZEN_BUILD_EXAMPLE}")#out: KAIZEN_BUILD_EXAMPLE_1: OFF## 方式一:修改option选项值(无效方式,请忽略)
option(${ PROJECT_NAME }_BUILD_EXAMPLE "构建示例模块" ON)
message(STATUS "KAIZEN_BUILD_EXAMPLE_2: ${KAIZEN_BUILD_EXAMPLE}")#out: KAIZEN_BUILD_EXAMPLE_2: OFF## 方式二:通过cache变量方式修改(正确方式,力推荐)
set(${ PROJECT_NAME }_BUILD_EXAMPLE ON CACHE BOOL "构建示例模块" FORCE)
message(STATUS "KAIZEN_BUILD_EXAMPLE_3: ${KAIZEN_BUILD_EXAMPLE}")#out: KAIZEN_BUILD_EXAMPLE_3 : ON
  • option()定义的变量本质上是缓存变量(CACHE变量)。当首次定义option()时,若缓存中不存在该变量,则创建并设置默认值;若已存在,则保留缓存值。
  • 使用set(... FORCE)会强制更新缓存中的值,无论之前是否存在或如何设置,从而确保变量被强制启用。

4.与 set(... CACHE BOOL ...) 的区别

虽然 option() 和 set(... CACHE BOOL ...) 都能创建可在 CMake 缓存中存储的布尔变量,但它们之间存在一些细微差别:

  • option()
    • 仅在变量未在缓存中定义时才会设置初始值。
    • 初始值参数是可选的,若不提供则默认为 OFF
    • 设计目的是专门创建用户可配置的选项。
  • set(... CACHE BOOL ...)
    • 无论变量是否已在缓存中定义,都会设置其值,除非使用 FORCE 参数。
    • 需要显式指定类型(如 BOOL)。
    • 更灵活,可用于创建各种类型的缓存变量。

5.示例代码

下面是一个完整的示例,展示了 option() 命令在实际项目中的应用:

cmake_minimum_required(VERSION 3.10)
project(MyProject VERSION 1.0.0)# 定义选项
option(BUILD_SHARED_LIBS "Build shared libraries" ON)
option(BUILD_TESTS "Build unit tests" ON)
option(ENABLE_PROFILING "Enable profiling support" OFF)# 设置 C++ 标准
set(CMAKE_CXX_STANDARD 17)# 根据选项构建库
if(BUILD_SHARED_LIBS)add_library(my_lib SHARED src/lib.cpp)
else()add_library(my_lib STATIC src/lib.cpp)
endif()# 根据选项添加编译定义
if(ENABLE_PROFILING)target_compile_definitions(my_lib PRIVATE ENABLE_PROFILING)target_link_libraries(my_lib PRIVATE profiling_library)
endif()# 根据选项构建测试
if(BUILD_TESTS)enable_testing()add_executable(my_tests tests/test.cpp)target_link_libraries(my_tests PRIVATE my_lib)add_test(NAME MyTests COMMAND my_tests)
endif()

通过使用 option() 命令,你的项目将变得更加灵活,用户可以根据自己的需求轻松调整编译选项。

6.总结

  • 确保在包含定义该option的子目录前调用set命令,或在项目顶层提前设置。
  • 避免在多个地方重复使用option()定义同一变量,以免引发冲突。
  • 使用有意义的选项名,选项名应简洁明了,能够清晰表达其用途,例如 BUILD_SHARED_LIBSENABLE_PROFILING 等。
  • 合理设置初始值,根据项目的常见使用场景来设置初始值,例如大多数库默认应构建为静态库还是共享库。
  • 避免硬编码选项值,尽量通过选项来控制编译行为,而不是在代码中硬编码条件判断。
http://www.xdnf.cn/news/8355.html

相关文章:

  • 代码讲解Java线程转换
  • 达梦数据库-学习-21-C 外部函数
  • 使用vscode MSVC CMake进行C++开发和Debug
  • 智能指针
  • 20250523在荣品的PRO-RK3566开发板的Android13下解决千兆网异常的问题【硬件部分】
  • 【算法系列】协同过滤算法
  • 通过vue-pdf和print-js实现PDF和图片在线预览
  • bi平台是什么意思?bi平台具体有什么作用?
  • Unreal渲染源码简读(一)RHI/Shader
  • MacOS Docker 启动 RustDesk Server OSS
  • 前端状态管理介绍(React状态管理)(通过管理状态来控制UI的呈现,确保数据的一致性和可维护性)(界面与数据分离、可预测性、组件化、响应式更新)
  • 高可用架构设计要点
  • 钉钉出口IP有哪些
  • linux strace调式定位系统问题
  • python学习day2:进制+码制+逻辑运算符
  • 云存储迁移遇瓶颈?大文件跨平台传输加速指南
  • hbuilder中h5转为小程序提交发布审核
  • Kafka Producer 如何实现Exactly Once消息传递语义
  • 用HTML5+JavaScript实现汉字转拼音工具
  • 使用KubeKey快速部署k8s v1.31.8集群
  • 车载诊断架构 --- 车载诊断有那些内容(下)
  • 技术服务业-首套运营商网络路由5G SA测试专网搭建完成并对外提供服务
  • threejs 四元数(Quaternion)和欧拉角(Euler)学习
  • 学习Raft共识算法基本原理
  • 了解Android studio 初学者零基础推荐(3)
  • Python版scorecardpy库woebin函数使用
  • 安全自动化与AI驱动防御
  • 《数据结构笔记三》:单链表(创建、插入、遍历、删除、释放内存等核心操作)
  • 教育行业课件共享难题:大文件分发效率优化方案
  • 广东省省考备考(第十八天5.23)—言语:语句排序题(听课后强化训练)