1.4k star 项目 CMakeTutorial 阅读和点评

1.4k star 项目 CMakeTutorial 阅读和点评

文章目录

  • 1.4k star 项目 CMakeTutorial 阅读和点评
    • 0. 概要
    • 1. CUDA 目录
    • 2. FindPackage 目录
    • 3. Installation 目录
    • 4. PackageManage 目录
    • 5. PythonExtension 目录
    • 6. ImportExternalProject 目录
    • 总结

0. 概要

在 github 搜索关键字 CMake, 靠前的结果中第一个中文仓库是 https://github.com/BrightXiaoHan/CMakeTutorial , 目前(2024.06.18) 有 1.2k, 今天阅读和点评这个项目。

相比于先前分析过的仓库,BrightXiaoHan 的 CMakeTutorial 仓库的优点:

  • 使用中文讲解,直白易懂
  • 提供了网页,可以在线阅读:
    https://brightxiaohan.github.io/CMakeTutorial/

缺点:

  • 只在 Linux 下实践, 没有 Windows 支持
  • 样例覆盖范围不够广泛, 只有6个目录
➜  CMakeTutorial git:(master) tree -L 1
.
├── CUDA
├── FindPackage
├── ImportExternalProject
├── Installation
├── LICENSE
├── PackageManage
├── PythonExtention
├── README.md
└── _config.yml

6 directories, 3 files

包含了6个目录,每个目录里是一个特定场景案例。 覆盖的场景显然不会太全, 不过每个提供的目录都还算精良, 下面逐一阅读点评。

1. CUDA 目录

在这里插入图片描述

作者提到从 cmake 3.9 开始, 原生支持了 CUDA C/C++. 也就是可以直接在 project() 中指定语言为 CUDA, 以及添加 .cu 文件到 target 中:

project(CUDA_MAT_MUL LANGUAGES CXX CUDA)
add_library(cudaMatMul cudaMatMul.cu cudaMatMul.h)
target_compile_features(cudaMatMul PUBLIC cxx_std_11)

比较遗憾的是,提供的引入 cublas 库的方式比较繁琐, 是参照了 CLblast - FindcuBLAS.cmake , 这仍然原本是包管理器应该做的事情, 转嫁负担到普通用户了。 此外, 提供的 FindcuBLAS.cmake 没有支持 Windows, 跨平台方面还需要改善, 不能直接拿去用。

在提供的示例代码方面, 作者比较用心了, 提供的是 CUDA 方面的入门例子: 矩阵乘法。 相比于简单的打印 hello world, MatMul 的代码更贴合实际场景, 体现了 C/C++ 和 cuda 在同一个文件中混合编译的用法。

2. FindPackage 目录

本节内容仍然是只能在 Linux 下使用, windows/macOS 没有适配, 是一个不足。

作者提到了 path_to_your_cmake/share/cmake-<version>/Modules 这个目录, 是 cmake 安装后提供的用于 find_package() 的包说明文件, 也就是众多的 Findxxx.cmake 文件、相关文件、子目录等。 这很难得, 前面看的 Cmake 高赞项目都没提到这点。

具体的案例, 作者使用了 curl 库为例子, 用 find_package 查询:

find_package(CURL)
add_executable(curltest curltest.cc)
if(CURL_FOUND)
    target_include_directories(clib PRIVATE ${CURL_INCLUDE_DIR})
    target_link_libraries(curltest ${CURL_LIBRARY})
else(CURL_FOUND)
    message(FATAL_ERROR ”CURL library not found”)
endif(CURL_FOUND)

不过, 上述代码显然是不够 modern, else(CURL_FOUND) 和 endif(CURL_FOUND) 可以把括号里留空。

作者还给出了通过自行配置, 让 find_package() 支持了 glog 库的导入。作者这里想表达的含义是, cmake 安装路径下的 share/cmake-/Modules 目录有时候没能覆盖你想找的库, 此时从源码编译你要找的库, 比如 glog, 并且假设你要找的库也是基于 cmake 构建的, 那么它(大概率)提供了 find_package() 的支持。

不过, glog 算是支持还算好的, 这种 find_package 支持, 全靠运气, 有些开源库作者就是没有写 cmake, 这时候需要另外想办法。这种情况可以细分, 对于是自己编写的代码, 比如基于 makefile 构建的库 libAdd, 作者提供了编写 FindAdd.cmake 的步骤。 这些步骤是可以用的, 但是只覆盖了 Linux, 并且目前(2024年)比较不推荐 FindAdd.cmake , 而是推荐 AddConfig.cmake 文件。这一节的内容稍微有点落伍了。

3. Installation 目录

这个目录介绍的是, 把自己编写的一个库 mymath,添加“安装”功能, 能安装它的头文件和库文件。 mymath 的代码简单而不失实用性, 是求整数和浮点数的和。 相比于 CMake 官方的 Tutorial 中求 sqrt 的代码, 对普通人更加友好。 介绍库的安装, 重点在于安装, 而不是把示例库本身的 cpp 代码搞得有点复杂。

这个例子写的还行:

install(TARGETS mymath
        EXPORT MyLibTargets
        LIBRARY DESTINATION lib  # 动态库安装路径
        ARCHIVE DESTINATION lib  # 静态库安装路径
        RUNTIME DESTINATION bin  # 可执行文件安装路径
        PUBLIC_HEADER DESTINATION include  # 头文件安装路径
)

几个不太行的地方:

  1. 关于默认安装位置, 可能4年前作者用的 CMake 版本, 在 windows 上是安装到 c:/Program Files/${PROJECT_NAME}, 也可能作者没有具体尝试仅仅是抄写了官方文档, 其实这个路径应该是 c:/Program Files (x86)/${PROJECT_NAME}.

  2. set_target_properties(mymath PROPERTIES PUBLIC_HEADER ${CMAKE_SOURCE_DIR}/include/mymath.h) 这句, 多余的。

  3. 关于 MyMathConfigVersion.cmakeMyMathConfig.cmake.in 文件:作者应该是仔细看了 CMake 官方文档, 整体看没有大问题。 主要的问题在于, 这东西写起来太繁琐(CMake的设计的问题)。 实际上如果是打算写一个包管理器, 那应该自动生成 MyMathConfig.cmake 文件, 尽可能减少手写那些又重复、又难写、又容易出错的写法, 例如:

include(CMakePackageConfigHelpers)
write_basic_package_version_file(
        MyMathConfigVersion.cmake
        VERSION ${PACKAGE_VERSION}
        COMPATIBILITY AnyNewerVersion  # 表示该函数库向下兼容
        )

4. PackageManage 目录

这个目录应该改名为 PackageManagement. 作者给出了两个包管理器的示例说明:

  • vcpkg
  • anaconda

具体到 CMakeLists.txt 文件, 和没有使用包管理器的写法是一样的。 可以说作者应该没有很深入的使用 vcpkg, anaconda 的话个人不太了解, 不评价。 对于 vcpkg, 在2024年显然应该补充 override 和 builtin-baseline 的设置例子, 因为这是锁定包的版本, 而不能锁定包版本的管理器显然都是不合格的。

这个目录其实几乎没用, 建议读者直接移步 vcpkg 或 conan 官方。

5. PythonExtension 目录

这节是 Python 扩展的用法, 作者使用了 Python 自带的头文件,而不是基于 Python 头文件的封装过的库如 PyBind11 等。

使用的具体代码, 是 Eigen 库的矩阵类, 封装了 random, ones, zeros 等函数。

cmake_minimum_required (VERSION 3.0)
project (PybindMatrix)

find_package (Eigen3 REQUIRED NO_MODULE)
find_package(PythonLibs REQUIRED)

include_directories(${PYTHON_INCLUDE_DIRS})
add_library(matrix python_matrix.cc)
target_compile_features(matrix PRIVATE cxx_std_11)
target_link_libraries (matrix ${PYTHON_LIBRARIES} Eigen3::Eigen)

这个例子其实有一定的实用价值, Eigen 自身有较好的计算性能, 如果是基于 Eigen 实现了某些功能, 或者打算用 Python 快速做一些实现, 基于已有的工程公共代码, 而公共代码是基于 Eigen 写的, 那么这个例子可以作为起始参考代码,能快速添加矩阵相关的计算。 当然了,抛开 C++ Eigen 库, 也可以用 python 的 pyquaternion 和 numpy, scipy 库, 但是那样的话, python 和 c++ 的库在二进制对齐方面可能就容易出岔子。

对于 Python 库的构建, 目前(2024年)看起来用 project.toml 比 setup.py 更好, 这是一个改进点。

6. ImportExternalProject 目录

这个目录介绍如何引入外部项目。 其实这仍然是包管理范畴的事情, 只不过现在可以添加的不一定是二进制库+头文件的传统“开发包”, 也可以是 header-only 库(其实就是源代码), 也可以是 头文件+源文件 的源码工程。

具体说来, 作者使用了 FetchContent, 这在网络良好的条件下确实是 OK 的:

# 添加第三方依赖包
include(FetchContent)
# FetchContent_MakeAvailable was not added until CMake 3.14
if(${CMAKE_VERSION} VERSION_LESS 3.14)
    include(add_FetchContent_MakeAvailable.cmake)
endif()

set(SPDLOG_GIT_TAG  v1.4.1)  # 指定版本
set(SPDLOG_GIT_URL  https://github.com/gabime/spdlog.git)  # 指定git仓库地址

FetchContent_Declare(
  spdlog
  GIT_REPOSITORY    ${SPDLOG_GIT_URL}
  GIT_TAG           ${SPDLOG_GIT_TAG}
)

FetchContent_MakeAvailable(spdlog)

但显然 FetchContent 是不够的, 作者这个目录也只能当玩具, 我认为需要补充的是:

  • add_subdirectory() 方式,使用外层目录下的子目录, 例如 fmtlib
  • add_subdirectory() 方式, 使用子目录或外层目录的子目录, 目录里的 CMakeLists.txt 中, 通过定义 IMPORTED 库来提供外部使用

总结

这个 1.2k star 的项目 CMakeTutorial , 在提供的内容中,CUDA 和 PythonExtension 在多年之后仍有较好的实用价值, Installation、 PackageManage, FindPackage, ImportExternalProject 则是凑合使用、 但根因在于 CMake 自身的设计, 初学者可以大概看看,但应寻求更好做法。 考虑到是中文写成, 易读性还是相当可以的。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/720926.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

ORA-12560: TNS:协议适配器错误

项目场景&#xff1a; 由于最近一直没有连接oracle&#xff0c;然后之前windows也是正常可以启动oracle&#xff0c;正常连接。无论是SQL Developer还是SQL PLUS命令&#xff0c;都能正常连接和操作。 问题描述 这两天刚好用SQL Developer工具连接&#xff0c;然后报错&#…

vue中通过自定义指令实现一个可拖拽,缩放的弹窗

效果 功能描述 按住头部可拖拽鼠标放到边框&#xff0c;可缩放多层重叠丰富的插槽&#xff0c;易于扩展 示例 指令代码 export const dragDialog {inserted: function (el, { value, minWidth 400, minHeight 200 }) {// 让弹窗居中let dialogHeight el.clientHeight ?…

和鲸101计划:以神经计算建模培训,助力北大学术人才培养

探索与求知&#xff0c;培养与传承。 让青年人更早地触摸到科学研究的前沿&#xff0c;便能吸引更多人才投身于学科建设。 11月4日&#xff0c;由北京大学信息处理实验室开展进行&#xff0c;北京大学心理与认知科学学院院长吴思教授及课题组成员授课的第二届神经计算建模及编…

C#心跳机制客户端

窗体&#xff08;richTextBox右显示聊天&#xff09; 步骤 点击链接按钮 tcpclient客户端步骤 1创建客户端对象 2连接服务器connect 3创建网络基础流发消息 .write发消息 4 创建网络基础流接消息 .read接消息 5 断开连接…

Spring注解----------@Deprecated

情景&#xff1a; 在我们开发过程中&#xff0c;有时候会遇到我们需要将几个类中的方法集中到一个类中&#xff0c;但是我们又不希望把我们的原来的类删掉&#xff08;就是单纯的不想删除&#xff0c;都是我写的代码我不想杀死我的结晶&#xff0c;不能说我写的是shi&#xff…

【C语言】--- 常见调试信息预处理器宏

在编程的艺术世界里&#xff0c;代码和灵感需要寻找到最佳的交融点&#xff0c;才能打造出令人为之惊叹的作品。而在这座秋知叶i博客的殿堂里&#xff0c;我们将共同追寻这种完美结合&#xff0c;为未来的世界留下属于我们的独特印记。 【C语言】--- 常见调试信息预处理器宏 开…

Multisim仿真之万用表、安捷伦万用表、信号发生器操作方法

1、XMM是安捷伦示波器 如下图所示&#xff0c;实物安捷伦的外围3个插孔对应于 XMM图标示波器的右侧3个引脚&#xff0c;上下一一对应 2、函数信号发生器XFG 如下图所示&#xff0c;COM就是GND&#xff0c;正负的意思就是相对于GND而言&#xff0c;有正负电压&#xff1b; 3、…

vivado PIP or SITE_PIP、PKGPIN_BYTEGROUP

PIP是Xilinx部件上用于路由连接或网络的设备对象。PIP 称为ARC的连接多路复用器可以编程为将一根电线连接到 另一个&#xff0c;从而将节点连接在一起&#xff0c;以形成中特定NET所需的路由 设计。 SITE_PIP&#xff0c;也称为路由BEL&#xff0c;是SITE内部的连接多路复用器&…

JavaFX按钮

当用户单击按钮时&#xff0c;JavaFX Button类可以触发事件。Button类扩展了Labeled类&#xff0c;可以显示文本&#xff0c;图像或两者都可以。 以下代码显示了如何向Button添加单击操作侦听器。 import javafx.application.Application; import javafx.event.ActionEvent; im…

【信息学奥赛】CSP-J/S初赛03 计算机网络与编程语言分类

第1节 计算机网络基础 1.1 网络的定义 所谓计算机网络&#xff0c;就是利用通信线路和设备&#xff0c;把分布在不同地理位置上的多台计算机连 接起来。计算机网络是现代通信技术与计算机技术相结合的产物。 网络中计算机与计算机之间的通信依靠协议进行。协议是计算机收、发…

Postman文件数据导入导出

前言 不同的接口测试工具如Postman、Apipost、Apifox创建的接口文档都是互通的&#xff0c;都可以互相兼容使用。我们就不需要在3个不同测试工具都去创建&#xff0c;只要在一个工具上创建&#xff0c;想要在其他接口测试工具上使用就运用导入和导出功能即可。 Postman、Apip…

2024.6.18 作业 xyt

今日作业&#xff1a; 1. 完善对话框&#xff0c;点击登录对话框&#xff0c;如果账号和密码匹配&#xff0c;则弹出信息对话框&#xff0c;给出提示”登录成功“&#xff0c;提供一个Ok按钮&#xff0c;用户点击Ok后&#xff0c;关闭登录界面&#xff0c;跳转到其他界面 如果…

【shell脚本速成】for 嵌套和和 if 高级用法

文章目录 一、for嵌套二、for与数组三、if高级用法3.1、条件符号使用双圆括号&#xff0c;可以在条件中植入数学表达式 if (())3.2、使用双方括号,可以在条件中使用通配符 四、简写if五、与文件存在与否的判断六、课后练习 &#x1f308;你好呀&#xff01;我是 山顶风景独好 &…

智能网站管理系统

智能网站管理系统&#xff0c;即智能化的网站管理工具&#xff0c;是为了提高网站管理效率和简化操作流程而开发的一种软件系统。它集合了各种先进的技术和功能&#xff0c;为网站管理员提供了一套强大而可靠的解决方案。 智能网站管理系统的核心功能是网站内容管理。传统的网站…

MySQL 基本语法讲解及示例(上)

第一节&#xff1a;MySQL的基本操作 1. 创建数据库 在 MySQL 中&#xff0c;创建数据库的步骤如下&#xff1a; 命令行操作 打开 MySQL 命令行客户端或连接到 MySQL 服务器。 输入以下命令创建一个数据库&#xff1a; CREATE DATABASE database_name;例如&#xff0c;创建一…

(资料收藏)王阳明传《知行合一》共74讲,王阳明知行合一音频讲解资料

今天给大家带来的不是软件&#xff0c;而是一份精神食粮——《知行合一》的教程福利。这可不是一般的教程&#xff0c;它关乎心灵&#xff0c;关乎智慧&#xff0c;关乎我们如何在纷繁复杂的世界中找到自己的位置。 咱们得聊聊王阳明&#xff0c;这位明代的大儒&#xff0c;他…

看完这篇文章你才能了解什么是大模型

引言 近年来&#xff0c;人工智能&#xff08;AI&#xff09;技术迅速崛起&#xff0c;成为全球科技领域的热门话题。大模型&#xff08;Large Language Model&#xff09;技术以其庞大的参数和复杂的结构&#xff0c;为AI提供了强大的计算和学习能力&#xff0c;推动着AI技术…

【proteus仿真】基于51单片机的秒表设计

【proteus仿真】基于51单片机的秒表设计 资料获取在文章结尾处&#xff01; 更多资料获取链接&#xff1a; https://docs.qq.com/sheet/DTExIc2dPUUJ5enZZ?tabBB08J2 1.资料内容 源码proteus仿真图 演示视频&#xff1a; 【proteus仿真】基于51单片机的秒表设计_哔哩哔…

​【数据结构与算法】冒泡排序:简单易懂的排序算法解析

&#x1f493; 博客主页&#xff1a;倔强的石头的CSDN主页 &#x1f4dd;Gitee主页&#xff1a;倔强的石头的gitee主页 ⏩ 文章专栏&#xff1a;《数据结构与算法》 期待您的关注 ​ 目录 一、引言 二、冒泡排序原理 &#x1f343;基本思想&#xff1a; &#x1f343;算法…

为什么传统 CNN 可能无法进行基于纹理的分类?

作者&#xff1a;Mayank Gubba、Mohammed Faisal、Trapti Kalra、Vijay Pandey 将纹理分析与深度学习结合使用对于在机器视觉任务中取得更好的结果起着重要作用。在第一篇博客中&#xff0c;我们讨论了“纹理”的基础知识、不同类型的纹理以及纹理分析在解决实际计算机视觉任务…