百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 热门文章 > 正文

【CMake】(13)流程控制

bigegpt 2025-01-18 17:30 3 浏览

条件判断

基本语法

条件判断的基本语法如下:

if(<condition>)
  <commands>
elseif(<condition>)
  <commands>
else()
  <commands>
endif()
  • if(<condition>):检查条件是否满足。如果满足,则执行随后的命令直到遇到 elseifelseendif
  • elseif(<condition>):可选。如果前面的 ifelseif的条件不满足,将检查这里的条件。可以有多个 elseif块。
  • else():可选。如果所有的 ifelseif条件都不满足,则执行 else块中的命令。
  • endif():结束条件判断块。

在CMake中,基本表达式用于 if语句中,以决定是否执行特定的代码块。这些表达式可以是常量、变量或字符串,CMake会根据这些表达式的值来判断条件为真(True)或假(False)。

判定表达式

当表达式的值为以下之一时,条件被认为是真(True):

  • 数值 1:代表真。
  • 字符串 ON:明确表示启用或真。
  • 字符串 YES:同样表示肯定或真。
  • 字符串 TRUE:布尔真值。
  • 字符串 Y:简短的肯定回答,表示真。
  • 非零数值:在大多数编程语言中,非零值通常被解释为真。
  • 非空字符串:任何非空的字符串都被视为真,除了下面将要提到的特定假值字符串外。

当表达式的值为以下之一时,条件被认为是假(False):

  • 数值 0:代表假。
  • 字符串 OFF:明确表示禁用或假。
  • 字符串 NO:表示否定或假。
  • 字符串 FALSE:布尔假值。
  • 字符串 N:简短的否定回答,表示假。
  • 字符串 IGNORE:有时用于特定的设置中,解释为假。
  • 字符串 NOTFOUND:特别在查找库或程序时,如果未找到,这个值表示假。
  • 空字符串:表示没有值,解释为假。
# 示例:变量设置为非零值
set(MY_VAR 42)
if(MY_VAR)
  message("MY_VAR is true")
endif()

# 示例:变量设置为假值字符串
set(MY_VAR "FALSE")
if(NOT MY_VAR)
  message("MY_VAR is false")
endif()

# 示例:使用未定义的变量
if(UNDEFINED_VAR)
  message("This will not be printed")
else()
  message("UNDEFINED_VAR is considered false")
endif()

逻辑判断

逻辑操作符 NOTANDOR允许你根据一个或多个条件来执行特定的代码块。这些操作符的行为与大多数编程语言中的逻辑操作符类似,非常直观。

NOT

NOT操作符用于取反一个条件的结果。如果原条件为真(True),则 NOT后的结果为假(False);如果原条件为假(False),则 NOT后的结果为真(True)。

set(VAR1 ON)

if(NOT VAR1)
  message("VAR1 is false")
else()
  message("VAR1 is true")
endif()

在这个例子中,因为 VAR1被设置为 ON(即真), NOT VAR1的结果为假,所以执行 else分支,输出"VAR1 is true"。

AND

AND操作符用于检查两个或更多条件是否都为真。只有当所有条件都为真时,整个 AND表达式的结果才为真(True);否则为假(False)。

set(VAR1 ON)
set(VAR2 OFF)

if(VAR1 AND VAR2)
  message("Both VAR1 and VAR2 are true")
else()
  message("At least one of VAR1 or VAR2 is false")
endif()

在这个例子中,因为 VAR1为真而 VAR2为假,整个条件表达式的结果为假,所以执行 else分支。

OR

OR操作符用于检查两个或更多条件中至少有一个是否为真。如果至少有一个条件为真,整个 OR表达式的结果就为真(True);只有当所有条件都为假时,结果才为假(False)。

set(VAR1 ON)
set(VAR2 OFF)

if(VAR1 OR VAR2)
  message("At least one of VAR1 or VAR2 is true")
else()
  message("Both VAR1 and VAR2 are false")
endif()

在这个例子中,因为至少有一个条件( VAR1)为真,所以整个条件表达式的结果为真,执行 if分支。

比较

数值比较

数值比较用于比较两个变量或字符串代表的数值,包括:

  • LESS:检查左侧的数值是否小于右侧的数值。
  • GREATER:检查左侧的数值是否大于右侧的数值。
  • EQUAL:检查两个数值是否相等。
  • LESS_EQUAL:检查左侧的数值是否小于或等于右侧的数值。
  • GREATER_EQUAL:检查左侧的数值是否大于或等于右侧的数值。
set(NUM1 10)
set(NUM2 20)

if(NUM1 LESS NUM2)
  message("NUM1 is less than NUM2")
endif()

字符串比较

字符串比较根据字符串的字典顺序来比较两个变量或字符串的大小,包括:

  • STRLESS:如果左侧字符串在字典顺序上小于右侧,返回True。
  • STRGREATER:如果左侧字符串在字典顺序上大于右侧,返回True。
  • STREQUAL:如果两个字符串相等,返回True。
  • STRLESS_EQUAL:如果左侧字符串小于等于右侧,返回True。
  • STRGREATER_EQUAL:如果左侧字符串大于等于右侧,返回True。
set(STR1 "apple")
set(STR2 "banana")

if(STR1 STRLESS STR2)
  message("apple comes before banana")
endif()

文件判断

判断文件或目录是否存在

  • EXISTS 操作用于检查指定路径的文件或目录是否存在。如果路径存在,条件为真(True),否则为假(False)。
if(EXISTS "${PROJECT_SOURCE_DIR}/myfile.txt")
  message("myfile.txt exists.")
else()
  message("myfile.txt does not exist.")
endif()

判断是否为目录

  • IS_DIRECTORY 用于检查给定的路径是否是一个目录。这要求提供的路径是绝对路径。
if(IS_DIRECTORY "${PROJECT_SOURCE_DIR}/mydir")
  message("mydir is a directory.")
else()
  message("mydir is not a directory.")
endif()

判断是否为软链接

  • IS_SYMLINK 操作用于检查指定的文件名是否是一个软链接。这同样要求文件名对应的路径是绝对路径。
if(IS_SYMLINK "/path/to/mylink")
  message("mylink is a symlink.")
else()
  message("mylink is not a symlink.")
endif()

判断是否为绝对路径

  • IS_ABSOLUTE 用于检查给定的路径是否是绝对路径。在Linux上,绝对路径以根目录( /)开始;在Windows上,它以盘符开始(如 C:/)。
if(IS_ABSOLUTE "/usr/local/bin")
  message("This is an absolute path.")
else()
  message("This is not an absolute path.")
endif()

其他

判断元素是否在列表中

CMake 3.3及更高版本支持 IN_LIST查询,这允许开发者检查一个变量或字符串是否在一个给定的列表中。

set(MY_LIST apple banana cherry)
set(MY_ITEM apple)

if(MY_ITEM IN_LIST MY_LIST)
  message("${MY_ITEM} is in the list.")
else()
  message("${MY_ITEM} is not in the list.")
endif()

比较两个路径是否相等

CMake 3.24及更高版本引入了 PATH_EQUAL,这对于比较两个路径是否相等非常有用,特别是在路径可能包含多余的分隔符时。 PATH_EQUAL会在比较前标准化路径,从而忽略多余的分隔符。

if("/path/to//directory" PATH_EQUAL "/path/to/directory")
  message("Paths are equal.")
else()
  message("Paths are not equal.")
endif()

STREQUAL相比, PATH_EQUAL在处理路径时更加智能,能够识别并处理路径中的冗余分隔符。这在跨平台开发中特别有用,因为不同操作系统的路径分隔符习惯可能不同。

以下示例演示了 PATH_EQUALSTREQUAL在处理包含多余分隔符的路径时的行为差异:

cmake_minimum_required(VERSION 3.26)
project(test)

if("/home//user///directory" PATH_EQUAL "/home/user/directory")
  message("Paths are equal using PATH_EQUAL.")
else()
  message("Paths are not equal using PATH_EQUAL.")
endif()

if("/home//user///directory" STREQUAL "/home/user/directory")
  message("Paths are equal using STREQUAL.")
else()
  message("Paths are not equal using STREQUAL.")
endif()

输出结果将展示 PATH_EQUAL成功地忽略了路径中的多余分隔符,而 STREQUAL则没有。

循环

有两种循环方式,分别是 foreachwhile

foreach

使用 foreach可以执行重复的任务,如设置变量、打印信息、或者根据列表中的每个项目执行特定的命令。

基本的 foreach循环语法如下:

foreach(loop_var IN ITEMS item1 item2 ... itemN)
  # 执行的命令
endforeach()
  • loop_var是循环变量,在每次循环中,它被设置为当前项的值。
  • ITEMS后面跟着的是一系列要遍历的项目。

遍历数值范围

  1. 单一终止值:
foreach(loop_var RANGE stop)
  # 执行的命令
endforeach()
  • 这里 loop_var从0遍历到 stop,包括 stop
  1. 指定起始、终止值(可选步长):
foreach(loop_var RANGE start stop [step])
  # 执行的命令
endforeach()
  • start:遍历开始的值。
  • stop:遍历结束的值。
  • step:(可选)遍历的步长,默认为1。

示例

当需要遍历一个从0开始到指定终止值的整数序列时,可以使用 RANGE关键字。

foreach(item RANGE 10)
    message(STATUS "当前遍历的值为: ${item}")
endforeach()

这将输出从0到10的数值,包括10。

示例2

如果需要从特定的起始值开始遍历,到特定的终止值结束,并且还可以指定步长(默认为1),可以使用增强版的 RANGE

foreach(item RANGE 10 30 2)
    message(STATUS "当前遍历的值为: ${item}")
endforeach()

这将从10开始,到30结束,步长为2,输出10, 12, 14, ..., 30。

遍历列表

foreach还可以遍历一个或多个列表,通过 IN LISTSITEMS关键字。

  1. 遍历单一或多个列表:
foreach(loop_var IN LISTS list1 [list2 ...])
  # 执行的命令
endforeach()
  • IN LISTS后面可以指定一个或多个列表变量。
  1. 遍历一系列项目:
foreach(loop_var IN ITEMS item1 [item2 ...])
  # 执行的命令
endforeach()
  • IN ITEMS后面跟着要直接遍历的项目序列。

示例:遍历列表或多个列表

  • 使用 IN LISTS直接遍历一个或多个列表变量:
set(WORD a b c d)
set(NAME ace sabo luffy)
foreach(item IN LISTS WORD NAME)
    message(STATUS "当前遍历的值为: ${item}")
endforeach()
  • 使用 IN ITEMS遍历一个或多个通过变量展开的列表:
set(WORD a b c "d e f")
set(NAME ace sabo luffy)
foreach(item IN ITEMS ${WORD} ${NAME})
    message(STATUS "当前遍历的值为: ${item}")
endforeach()

遍历多个列表(CMake 3.17及以上)

foreach(loop_var1 loop_var2 ... IN ZIP_LISTS list1 list2 ...)
  # 执行的命令
endforeach()
  • IN ZIP_LISTS允许同时遍历多个列表, loop_var1loop_var2等变量分别对应每个列表中当前位置的元素。

示例:ZIP_LISTS

在CMake 3.17及以上版本中, foreach命令引入了 IN ZIP_LISTS,这允许你同时遍历多个列表,并在每次迭代中从每个列表中取出相应位置的元素。

list(APPEND WORD hello world "hello world")
list(APPEND NAME ace sabo luffy zoro sanji)
foreach(item1 item2 IN ZIP_LISTS WORD NAME)
    message(STATUS "当前遍历的值为: item1 = ${item1}, item2=${item2}")
endforeach()

这种方式特别适用于需要同时处理多个相关列表的情况,如同时遍历文件名列表和对应的目标名列表。

while

foreach循环相比, while循环更加灵活,因为它允许你在每次循环结束时基于复杂的逻辑来更新循环条件。这使得 while循环特别适合于处理那些在循环开始前不能确定迭代次数的情况。

基本语法:

while(<condition>)
    <commands>
endwhile()
  • <condition>:这是循环继续执行的条件。只要条件评估为真(即,非0或非空字符串,或特定的CMake真值,如 ONTRUE等),循环就会继续。
  • <commands>:在每次循环迭代中执行的命令集。

示例:遍历列表并逐个移除元素

在你提供的示例中, while循环用于遍历一个列表,并在每次迭代中从列表的头部移除一个元素,直到列表为空:

cmake_minimum_required(VERSION 3.5)
project(test)

# 创建一个列表
set(NAME luffy sanji zoro nami robin)

# 获取列表长度
list(LENGTH NAME LEN)

# 循环直到列表为空
while(${LEN} GREATER 0)
    message(STATUS "names = ${NAME}")
    # 从列表头部移除一个元素
    list(POP_FRONT NAME)
    # 更新列表长度
    list(LENGTH NAME LEN)
endwhile()

每次循环迭代都会更新列表 NAME,并重新计算其长度 LEN。循环继续执行直到 LEN为0,即列表中不再有元素。

相关推荐

10w qps缓存数据库——Redis(redis缓存调优)

一、Redis数据库介绍:Redis:非关系型缓存数据库nosql:非关系型数据库没有表,没有表与表之间的关系,更不存在外键存储数据的形式为key:values的形式c语言写的服务(监听端口),用来存...

Redis系列专题4--Redis配置参数详解

本文基于windowsX64,3.2.100版本讲解,不同版本默认配置参数不同在Redis中,Redis的根目录中有一个配置文件(redis.conf,windows下为redis.windows....

开源一夏 | 23 张图,4500 字从入门到精通解释 Redis

redis是目前出场率最高的NoSQL数据库,同时也是一个开源的数据结构存储系统,在缓存、数据库、消息处理等场景使用的非常多,本文瑞哥就带着大家用一篇文章入门这个强大的开源数据库——Redis。...

redis的简单与集群搭建(redis建立集群)

Redis是什么?是开源免费用c语言编写的单线程高性能的(key-value形式)内存数据库,基于内存运行并支持持久化的nosql数据库作用主要用来做缓存,单不仅仅是做缓存,比如:redis的计数器生...

推荐几个好用Redis图形化客户端工具

RedisPlushttps://gitee.com/MaxBill/RedisPlusRedisPlus是为Redis可视化管理开发的一款开源免费的桌面客户端软件,支持Windows、Linux...

关于Redis在windows上运行及fork函数问题

Redis在将数据库进行持久化操作时,需要fork一个进程,但是windows并不支持fork,导致在持久化操作期间,Redis必须阻塞所有的客户端直至持久化操作完成。微软的一些工程师花费时间在解决在...

你必须懂的Redis十大应用场景(redis常见应用场景)

Redis作为一款高性能的键值存储数据库,在互联网业务中有着广泛的应用。今天,我们就来详细盘点一下Redis的十大常用业务场景,并附上Golang的示例代码和简图,帮助大家更好地理解和应用Redis。...

极简Redis配置(redis的配置)

一、概述Redis的配置文件位于Redis安装目录下,文件名为redis.conf(Windows名为redis.windows.conf,linux下的是redis.conf)你可以通过C...

什么是redis,怎么启动及如何压测

从今天起咱们一起来学习一下关于“redis监控与调优”的内容。一、Redis介绍Redis是一种高级key-value数据库。它跟memcached类似,不过数据可以持久化,而且支持的数据类型很丰富。...

一款全新Redis UI可视化管理工具,支持WebUI和桌面——P3X Redis UI

介绍P3XRedisUI这是一个非常实用的RedisGUI,提供响应式WebUI访问或作为桌面应用程序使用,桌面端是跨平台的,而且完美支持中文界面。Githubhttps://github....

windows系统的服务器快速部署java项目环境地址

1、mysql:https://dev.mysql.com/downloads/mysql/(msi安装包)2、redis:https://github.com/tporadowski/redis/r...

window11 下 redis 下载与安装(windows安装redis客户端)

#热爱编程是一种怎样的体验#window11下redis下载与安装1)各个版本redis下载(windows)https://github.com/MicrosoftArchive/r...

一款轻量级的Redis客户端工具,贼好用!

使用命令行来操作Redis是一件非常麻烦的事情,我们一般会选用客户端工具来操作Redis。今天给大家分享一款好用的Redis客户端工具TinyRDM,它的界面清新又优雅,希望对大家有所帮助!简介Ti...

一个.NET开发且功能强大的Windows远程控制系统

我们致力于探索、分享和推荐最新的实用技术栈、开源项目、框架和实用工具。每天都有新鲜的开源资讯等待你的发现!项目介绍SiMayRemoteMonitorOS是一个基于Windows的远程控制系统,完...

Redis客户端工具详解(4款主流工具)

大家好,我是mikechen。Redis是大型架构的基石,也是大厂最爱考察内容,今天就给大家重点详解4款Redis工具@mikechen本篇已收于mikechen原创超30万字《阿里架构师进阶专题合集...