CMake 函數和宏

CMake 函數

CMake 函數定義語法如下, 其中 name 為函數名, <arg1> 為參數名, <commands> 為函數體. 函數定義后, 可以通過 name 調用函數. 函數名允許字母數字下劃線, 不區分大小寫.

function(name [<arg1> ...])<commands>
endfunction()

如下的樣例定義了一個函數fun, 不帶任何參數.

function(fun)message("Hello, World!")
endfunction()# 調用函數
fun()
FUN()
Fun()
cmake_language(CALL fun)# 攜帶了參數, 參數被函數忽略
FUN(A B C)

函數參數

CMake 將參數分為如下幾種類型:

  1. 命名參數, 按照攜帶值的數量進一步分為:
    1. option類型, 不攜帶任何值, 如果存在則視為真, 不存在視為假. 比如ENABLE_TESTS指定是否編譯測試.
    2. single類型, 攜帶一個值. 比如參數OUTPUT指定一個目標文件.
    3. multi類型, 攜帶多個值. 比如參數SOURCE指定多個源文件.
  2. 未命名的參數.

CMake 對于每個函數都自動定義了如下三個變量:

  1. ARGC: 函數參數個數.
  2. ARGV: 函數參數列表. 包含命名參數和未命名參數.
  3. ARGN: 只包含未命名參數.

我們先看一下ARGN的使用場景:

function(add_gtest targetName)add_executable(${targetName} ${ARGN})target_link_libraries(${targetName} PRIVATE GTest::gtest)add_test(NAME ${targetName} COMMAND ${targetName})
endfunction()# 使用方式
add_gtest(test1 test1.cpp)
add_gtest(test2 test2.cpp util.cpp)

參數解析

CMake 使用cmake_parse_arguments來解函數參數, 這個函數有兩種調用方式

cmake_parse_arguments(<prefix> <options> <one_value_keywords><multi_value_keywords> <args>...)cmake_parse_arguments(PARSE_ARGV <N> <prefix> <options><one_value_keywords> <multi_value_keywords>)

第二種方式是在 3.7 版本引入的, 并且不能再宏中使用. 二者的區別在于PARSE_ARGV指定了參數列表的起始位置, 這在一些嵌套的函數參數傳遞中有用.

function(fun)set(options ENABLE_A ENABLE_B ENABLE_C)set(single OUTPUT_NAME)set(multi DEPENDS SOURCES)cmake_parse_arguments(arg "${options}" "${single}" "${multi}" ${ARGN})foreach(opt IN LISTS options)if(arg_${opt})message(STATUS "${opt} is set")endif()endforeach()if (arg_OUTPUT_NAME)message(STATUS "OUTPUT_NAME=${arg_OUTPUT_NAME}")endif()foreach(key IN LISTS multi)if (arg_${key})message(STATUS "${key}=${arg_${key}}")endif()endforeach()
endfunction()# 調用函數
fun(ENABLE_AOUTPUT_NAME "output.exe"DEPENDS "lib-a" "lib-b" "lib-c"SOURCES s1.cpp s2.cpp s3.cpp
)

輸出:

-- ENABLE_A is set
-- OUTPUT_NAME=output.exe
-- DEPENDS=lib-a;lib-b;lib-c
-- SOURCES=s1.cpp;s2.cpp;s3.cpp

設置返回值

從 CMake 3.25 開始, CMake 支持return語句中設置返回值. 注意此時需要設置 CMake Policy CMP0140NEW.

cmake_minimum_required(VERSION 3.25)
cmake_policy(SET CMP0140 NEW)function(getVal retValName)set(${retValName} "Hello, World!")return (PROPAGATE ${retValName})
endfunction()getVal(ret1)
message(STATUS "ret1=${ret1}") # 輸出 -- ret1=Hello, World!

而在以前的版本中, 一般是通過set變量存在于父級的作用域達到返回值目的.

function(getValOld retValueName)set(${retValueName} "Glad to see you" PARENT_SCOPE)
endfunction()getValOld(ret2)
message(STATUS "ret2=${ret2}")

常見錯誤

  1. 函數重復定義. 當使用 function()macro() 定義一個新命令時, 如果已經存在同名的命令, CMake 有一個未記錄的行為: 舊命令會以原名稱加下劃線的形式繼續可用. 無論舊名稱是內置命令, 還是自定義函數或宏, 都是如此. 了解這一行為的開發者有時會試圖利用它來創建現有命令的包裝器,
function(fun)message("call 1")
endfunction()function(fun)message("call 2")_fun()
endfunction()function(fun)message("call 3")_fun()
endfunction()fun()

這個函數將會無限循環, 并最終導致棧溢出.

  1. 第二次定義的時候, _fun指向第一個定義的函數, 此時還是可以正常工作的.
  2. 第三次定義的時候, _fun已經指向了第二個定義的函數, 而第二個定義的函數中又調用了_fun, 因此會無限循環.

CMake 宏

CMake 宏的定義方式與函數定義方式相同, 定義語法與函數定義語法相同.

macro(name [arg1 [arg2 [...]]])# command list...
endmacro()

宏在調用之后就是被粘貼到調用的位置, 宏不會產生一個新的作用域. 跟 C/C++中的#define類似, 其實本質上就是做的文本替換.

專欄目錄

  • 快速上手
  • 最佳實踐
  • CMake基礎: 變量
  • CMake基礎: 控制流
  • CMake基礎: 函數和宏

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/bicheng/74415.shtml
繁體地址,請注明出處:http://hk.pswp.cn/bicheng/74415.shtml
英文地址,請注明出處:http://en.pswp.cn/bicheng/74415.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

【QA】Qt有哪些迭代器模式的應用?

在 Qt/C 中&#xff0c;迭代器模式的設計主要分為 標準 C 風格 和 Qt 框架特有風格&#xff0c;以下結合代碼詳細說明兩種實現方式的關鍵設計及其應用場景&#xff1a; 一、Qt 框架中的迭代器模式設計 Qt 提供了兩種迭代器風格&#xff1a;Java 風格&#xff08;顯式迭代器&am…

Mysql表的簡單操作

&#x1f3dd;?專欄&#xff1a;Mysql_貓咪-9527的博客-CSDN博客 &#x1f305;主頁&#xff1a;貓咪-9527-CSDN博客 “欲窮千里目&#xff0c;更上一層樓。會當凌絕頂&#xff0c;一覽眾山小。” 目錄 3.1 創建表 3.2 查看表結構 3.3 修改表 1. 添加字段 2. 修改字段 3…

【云馨AI-大模型】自動化部署Dify 1.1.2,無需科學上網,Linux環境輕松實現,附Docker離線安裝等

Dify介紹 官網&#xff1a;https://dify.ai/zh生成式 AI 應用創新引擎開源的 LLM 應用開發平臺。提供從 Agent 構建到 AI workflow 編排、RAG 檢索、模型管理等能力&#xff0c;輕松構建和運營生成式 AI 原生應用。 Dify安裝腳本 目錄創建 mkdir -p /data/yunxinai &&a…

WordPress上傳圖片時顯示“未提供數據”錯誤

在WordPress中上傳圖片時顯示“未提供數據”的錯誤&#xff0c;通常是由多種原因引起的&#xff0c;以下是一些常見的問題及其解決方法&#xff1a; 1. 文件權限問題 WordPress需要正確的文件和目錄權限才能正常上傳圖片。如果權限設置不正確&#xff0c;可能會導致無法上傳圖…

python3面試題20個(python web篇)

更多內容請見: python3案例和總結-專欄介紹和目錄 文章目錄 1.python asyncio的原理?2.對Flask藍圖(Blueprint)的理解?3.Flask 和 Django 路由映射的區別?4.什么是wsgi,uwsgi,uWSGI?5.Django、Flask、Tornado的對比?6.CORS 和 CSRF的區別?7.Session,Cookie,JWT的理解8.簡…

RedisTemplate和RedissonClient適用的場景有什么不同

在 Spring Boot 項目中&#xff0c;RedisTemplate 和 RedissonClient 分別針對不同的使用場景設計&#xff0c;以下是它們的核心區別和適用場景分析&#xff1a; 一、RedisTemplate&#xff08;Spring Data Redis&#xff09; 定位 Spring 官方提供的 Redis 操作工具&#xf…

人臉表情識別系統分享(基于深度學習+OpenCV+PyQt5)

最近終于把畢業大論文忙完了&#xff0c;眾所周知碩士大論文需要有三個工作點&#xff0c;表情識別領域的第三個工作點一般是做一個表情識別系統出來&#xff0c;如下圖所示。 這里分享一下這個表情識別系統&#xff1a; 采用 深度學習OpenCVPyQt5 構建&#xff0c;主要功能包…

GitHub供應鏈攻擊事件:Coinbase遭襲,218個倉庫暴露,CI/CD密鑰泄露

此次供應鏈攻擊涉及GitHub Action "tj-actions/changed-files"&#xff0c;最初是針對Coinbase的一個開源項目的高度定向攻擊&#xff0c;隨后演變為范圍更廣的威脅。 攻擊過程與影響 Palo Alto Networks Unit 42在一份報告中指出&#xff1a;“攻擊載荷主要針對其…

Redis 核心源碼解析:從設計哲學到企業級應用實踐

一、Redis 的核心設計哲學 Redis 的成功源于其 「用內存換時間」 的核心理念&#xff0c;圍繞以下三個核心原則構建&#xff1a; 極簡主義&#xff1a;單線程模型避免鎖競爭&#xff0c;代碼保持高度內聚。 性能至上&#xff1a;所有數據常駐內存&#xff0c;網絡層采用事件驅…

GZCTF平臺搭建及題目上傳

前言 我用手里的Ubuntu虛擬機搭建的&#xff0c;大家根據自己的實際情況來吧 安裝及部署 首先&#xff0c;你的虛擬機需要有Docker和Docker-Compose&#xff0c;前者可以看我之前的文章&#xff0c;另外一個可以輸入下面的命令安裝&#xff0c;注意先獲取管理員權限&#xff…

Pycharm社區版創建Flask項目詳解

一、創建工程項目 二、配置工程目錄 新建的空項目下創建目錄。 1、新建app.py文件 2、app.py代碼如下&#xff1a; from flask import Flask, render_templateapp Flask(__name__)app.route("/") def root():"""主頁:return: Index.html"&qu…

CentOS 7 64位安裝Docker

以下是在已有的 CentOS 7 64 位虛擬機上安裝 Docker 并配置華為鏡像源的詳細步驟&#xff1a; 1. 備份原有 Yum 源&#xff08;可選&#xff0c;建議操作&#xff09; # 備份原有倉庫文件 sudo mv /etc/yum.repos.d/CentOS-Base.repo /etc/yum.repos.d/CentOS-Base.repo.backu…

運動仿真——phased.Platform

在雷達仿真過程中&#xff0c;運動仿真的必要性&#xff0c;以及運動仿真可以實現哪些功能&#xff0c;在matlab對應的user guide中已經講的很清楚了&#xff0c;這里不再贅述。 本文主要介紹phased.Platform的一些“坑”&#xff0c;和典型的用法。 第一坑&#xff1a;系統對…

緩存刪除三級補償方案:延遲隊列+消息隊列+定時任務兜底

問題背景: 在 Cache-Aside 模式中&#xff0c;更新數據庫后刪除緩存失敗會導致數據不一致。本文提供工業級三級補償方案&#xff0c;實現最終一致性保障。 整體架構: 更新操作觸發 → 一級延遲隊列 → 二級消息隊列 → 三級定時任務方案實現: 一、第一級補償&#xff1a;延遲隊…

從零開始實現 C++ TinyWebServer 數據庫連接池 SqlConnectPool詳解

文章目錄 數據庫連接池是什么&#xff1f;Web Server 中為什么需要數據庫連接池&#xff1f;SqlConnectPool 成員變量實現 Init() 函數實現 ClosePool() 函數SqlConnectRAII 類SqlConnectPool 代碼SqlConnectPool 測試 從零開始實現 C TinyWebServer 項目總覽 項目源碼 數據庫連…

C++題目

1、內存管理 1.內存模型 棧:在執行函數時&#xff0c;函數內局部變量的存儲單元都可以在棧上創建&#xff0c;函數執行結束時這些存儲單元自動被釋放。 堆&#xff1a;就是那些由new分配的內存塊&#xff0c;其釋放由程序員控制&#xff08;一個new對應一個delete&#xff09…

天地圖InfoWindow插入React自定義組件

截至2025年03月21日天地圖的Marker不支持添加Label; 同時Label和Icon是不支持自定義HTMLElement只支持String&#xff1b;目前只有InfoWindow支持自定義HTMLElement; 效果圖 React核心api import ReactDOM from react-dom/client const content document.createElement(div);…

Java并發編程面試匯總

Java并發編程 一、 基礎概念1. 進程與線程的區別是什么&#xff1f;2. 創建線程的幾種方式&#xff1f;3. 線程的生命周期&#xff08;狀態&#xff09;有哪些&#xff1f;4. 什么是守護線程&#xff08;Daemon Thread&#xff09;&#xff1f;5. 線程優先級&#xff08;Priori…

【STM32】第一個工程的創建

目錄 1、獲取 KEIL5 安裝包2、開始安裝 KEIL52.1、 激活2.2、安裝DFP庫 3、工程創建4、搭建框架5、開始編寫代碼 1、獲取 KEIL5 安裝包 要想獲得 KEIL5 的安裝包&#xff0c;在百度里面搜索“KEIL5 下載”即可找到很多網友提供的下載文件&#xff0c;或者到 KEIL 的官網下載&a…

動態規劃~01背包問題

01背包問題 經典的0 - 1背包問題的解決方案。 二維數組的版本 代碼功能概述 0 - 1背包問題指的是有 n 個物品和一個容量為 m 的背包&#xff0c;每個物品有對應的體積 v[i] 和價值 w[i]&#xff0c;需要從這些物品里挑選若干個放入背包&#xff0c;讓背包內物品的總價值達到最…