CMakeLists.txt基础指令与cmake-gui生成VS项目的步骤
简介
本博客主要介绍cmake的基本指令,同时,很多使用Visual Studio小白从Gitbub下载项目源码后,看到CMakeLists.txt,不知道如何使用Visual Studio编译源码;针对以上问题,做一下简单操作与解释,方便小白入门,大佬就没必要继续看啦…
CMakeLists.txt的作用
基本来说,CMakeLists.txt的主要作用有:
- 告诉编译器,源码之间的目录关系;
- 告诉编译器,该怎么编译链接源码;
第一种,就是告诉编译器,我要include的文件放在了哪里,我要的静态库,动态库放在了哪个文件夹内;第二种,主要是理解项目源码的编译过程,这方面自行百度了解。接下来用几个基本操作说明;
示例一
源码目录一结构
先简单的写一个cpp文件,在include文件夹下放置test.h和test.cpp文件;LearnCmake.cpp主要包含main函数,test.h和test.cpp的主要内容是:
//test.h的内容
#pragma once
#include <iostream>
void fun();//test.cpp的内容
#include "test.h"
void fun()
{std::cout << "测试函数" << std::endl;
}
LearnCmake.cpp的内容是:
#include <iostream>
#include "test.h"
using namespace std;
int main()
{fun();cout << "输出结果" << endl;system("pause");return 0;
}
实操的话,源码目录我已上传:
cmake示例
这里并没有使用编译器生成项目文件,仅仅是编写了源码;源码的目录结构是:
MyProject/|── CMakeLists.txt|── LearnCmake.cpp|── include/| |──test.h| └──test.cpp
在main函数的同级目录下定义了CMakeLists.txt文件,内容是:
#所需的最低版本
cmake_minimum_required(VERSION 3.0)#本CMakeLists.txt的project名称,用于生成learnCmake.sln
project(learnCmake)#设置生成文件的路径,这将会把生成的可执行文件放置在 ${CMAKE_BINARY_DIR}/bin 文件夹中
set(CMAKE_RUNTIME_OUTPUT_DIRECTORY ${CMAKE_BINARY_DIR}/bin)#设置include文件夹路径,可以理解为附加包含目录
include_directories(include)#先添加test.cpp文件到一个名为test_lib的库
add_library(test_lib include/test.cpp)#将.cpp/.c/.cc文件生成可执行文件,a就是a.exe
add_executable(a LearnCMake.cpp)# 将test_lib链接到a.exe,一定是先add_executable,再target_link_libraries
target_link_libraries(a test_lib)
注意以上指令的顺序,首先,先将test.cpp编译成一个静态库,同时生成一个a.exe,再将静态库链接到a.exe内;顺序不能错了,不能先链接到exe(因为此时exe还未生成),再生成exe;这样从cmake里就能看到源码编译的过程,也是上面我说的第二条,也能更好地理解编译的过程;
同时,add_library里需要将一个cpp文件编译成lib文件,这里的include/test.cpp,include就是CMakeLists.txt同一级目录,源码项目以CMakeLists.txt为起点,这样便说明了test.cpp所在源码的相对位置,这就是上面我说的第一条;
使用cmake-gui将源码目录一生成Visual Studio项目
项目源码写好之后,不同的平台,不同的编译器会根据CMakeLists.txt里的指令生成各自的项目文件,在Windows平台上,可以使用cmake-gui根据CMakeLists.txt里的指令,生成你电脑上安装好的Visual Studio项目。

这里介绍使用camke-gui调用刚才的源码里的CMakeLists.txt:


可以看到,在2步骤下的文件目录下已经生成了Visiual Studio的项目,项目名称是:learnCmake.sln,这与上面cmake指令project(learnCmake)
一致。

打开项目,可以看到解决方案有:

其中,ALL_BUILD用于编译整个项目的工程。ALL_BUILD相当于makefile里面的默认目标,构建整个项目,但不包括install和单元测试等。ZERO_CHECK监视CMakeLists.txt,如果CMakeLists.txt发生变化,则告诉编译器重新构建整个工程环境。ZERO_CHECK是首先执行的构建目标,会检查生成出的VS项目相比CMakeLists.txt是否过期,如果过期会首先重新生成VS项目。所有其它目标都会依赖这个ZERO_CHECK,于是构建别的目标都会先走一下ZERO_CHECK,保证了所生成项目的即时性。可以直接将a设定启动项目。右键a项目,属性内可以看到


已经按照cmake的指令,将附加包含目录和静态依赖库文件写好了,这就是对应上面的第一和第二条。
直接运行a项目,便可以生成exe了,结果是:

示例二
有的源码项目内会出现多个CMakeLists.txt,比如这样的目录结构:
MyProject/
|── CMakeLists.txt
|── main.cpp
|── include/
| |──test.h
| |──test.cpp
| └──CMakeLists.txt
和main同一级的CMakeLists.txt的内容如下:
#所需的最低版本
cmake_minimum_required(VERSION 3.0)#本CMakeLists.txt的project名称,生成learnCmake.sln
project(learnCmake)#设置生成文件的路径,这将会把生成的可执行文件放置在 ${CMAKE_BINARY_DIR}/bin 文件夹中
set(CMAKE_RUNTIME_OUTPUT_DIRECTORY ${CMAKE_BINARY_DIR}/bin)#设置include文件夹路径,可以理解为附加包含目录
include_directories(include)#添加执行子目录的CMakeLists.txt
add_subdirectory(include)#连接静态库
link_libraries(test)#将.cpp/.c/.cc文件生成可执行文件,a就是a.exe
add_executable(a LearnCMake.cpp)
和test.cpp同一级的CMakeLists.txt的内容如下:
#搜索当前目录下所有源文件
aux_source_directory(./ SRC)#编译静态库文件
add_library(test ${SRC})
可以这样理解,先将test.cpp编译成test.lib,再将静态库编译到a.exe内;
示例三
源码项目内需要链接别人的lib文件,并且需要拷贝dll文件,比如这样的目录结构:
MyProject/
|── CMakeLists.txt
|── LearnCmake.cpp
|── lib/
| └──testDll.lib
|── include/
| |──test.h
| |──test.cpp
| └──testDll.h
|── dll/
| └──testDll.dll
CMake指令需要这样写:
#所需的最低版本
cmake_minimum_required(VERSION 3.0)#本CMakeLists.txt的project名称,生成learnCmake.sln
project(learnCmake)#生成的类型,Debug后不能缺少分号
set(CMAKE_CONFIGURATION_TYPES "Debug;")#设置生成文件的路径,这将会把生成的可执行文件放置在 ${CMAKE_BINARY_DIR}/bin 文件夹中
set(CMAKE_RUNTIME_OUTPUT_DIRECTORY ${CMAKE_BINARY_DIR}/bin)#设置include文件夹路径,可以理解为附加包含目录
include_directories(include)#先添加test.cpp文件到一个名为test_lib的库
add_library(test_lib include/test.cpp)#将.cpp/.c/.cc文件生成可执行文件,a就是a.exe
add_executable(a LearnCMake.cpp)# 将test_lib链接到a.exe,一定是先add_executable,再target_link_libraries
target_link_libraries(a test_lib)#这里相当于VS的附加依赖项,将testDll.lib链接到a.exe,注意这里带后缀名
target_link_libraries(a ${CMAKE_CURRENT_SOURCE_DIR}/lib/testDll.lib)# 将testDll.dll 复制到a.exe所在的目录,这里仅仅起拷贝的作用
add_custom_command(TARGET a POST_BUILDCOMMAND ${CMAKE_COMMAND} -E copy_if_different"${CMAKE_CURRENT_SOURCE_DIR}/dll/testDll.dll"$<TARGET_FILE_DIR:a>
)
Github项目源码通过cmake-gui生成VS项目示例
这里下载一个github上的一个C++热门入门项目,项目链接为:
MyTinySTL
可以看到项目目录中的CMakeLists.txt

使用camke-gui生成VS项目后,可以看到sln文件

打开项目文件,将stltest设为启动项目,编译后可以看到

结尾
关于cmake的指令比我这篇博客讲的更多,这里仅入门讲解,还是推荐小白下载项目目录后亲自测试一遍,可能心得体会更多;
相关文章:
CMakeLists.txt基础指令与cmake-gui生成VS项目的步骤
简介 本博客主要介绍cmake的基本指令,同时,很多使用Visual Studio小白从Gitbub下载项目源码后,看到CMakeLists.txt,不知道如何使用Visual Studio编译源码;针对以上问题,做一下简单操作与解释,方…...
IT应用运维最常用指标
可用性(Availability) 系统或服务在特定时间范围内可用的百分比。 计算方式:(总时间 - 不可用时间)/ 总时间 * 100%。 参考值:99.9%。 应用范围:应用系统、网络设备。 故障率(Fa…...
Go中各种newreader和newbuffer的使用
一、bytes.NewBuffer和bytes.NewReader func main() {var byteArr []bytebuf : bytes.NewBuffer(byteArr)buf.Write([]byte("今天不错"))fmt.Println(buf.String()) }package mainimport ("bytes""fmt" )func main() {data : []byte("路多…...
visual studio 如何建立 C 语言项目
安装这个 模块。 新建 空项目 创建完成 写demo 点击运行:...
app小程序定制开发的优势|企业软件网站建设
app小程序定制开发的优势|企业软件网站建设 小程序定制开发是目前互联网行业中备受关注的领域之一。随着智能手机的普及和移动互联网的迅猛发展,越来越多的企业和个人开始重视小程序的潜力,并积极寻求定制开发的服务。那么,为什么小程序定制开…...
物联网AI MicroPython学习之语法 WDT看门狗外设
学物联网,来万物简单IoT物联网!! WDT 介绍 模块功能: 看门狗WDT(WatchDog Timer)外设驱动模块 接口说明 WDT - 构建WDT对象 函数原型:WDT(timeout)参数说明: 参数类型必选参数?…...
JVM线程的几种状态
1.New 新建的线程,线程还没启动。 2.Runnable 线程正在运行或者等待操作系统中的其他资源,例如线程运行过程中,系统分配资源给其他操作,此时这个线程还是Runnable状态,可以理解为可运行的线程。 3.Blocked 阻塞状…...
基于单片机停车场环境监测系统仿真设计
**单片机设计介绍, 基于单片机停车场环境监测系统仿真设计 文章目录 一 概要二、功能设计设计思路 三、 软件设计原理图 五、 程序六、 文章目录 一 概要 基于单片机的停车场环境监测系统是一种利用单片机技术实现环境监测和数据处理的系统。它可以感知停车场的温湿…...
每日一题:LeetCode-589.N叉树的前序遍历
每日一题系列(day 01) 前言: 🌈 🌈 🌈 🌈 🌈 🌈 🌈 🌈 🌈 🌈 🌈 🌈 🌈 🔎…...
PTA 7-2 简单计算器
7-2 简单计算器 分数 20 全屏浏览题目 作者 张彤彧 单位 浙江大学 模拟简单运算器的工作。假设计算器只能进行加减乘除运算,运算数和结果都是整数,四种运算符的优先级相同,按从左到右的顺序计算。 输入格式: 输入在一行中给出一个四则运…...
9、鸿蒙应用桌面图标外观和国际化
一、项目资源目录 项目下的resoueces目录为资源配置目录,其中base为基础配置,即在任何语言环境下都会加载的资源, color.json:用于配置颜色,如页面的背景和文字的颜色。 string.json:用于设置文字&#…...
oracle rac 19c修改不同网段public ip
客户需求将才搭建的oracle 19.19数据库从192.168.168.0网段调整到192.168.213网段 1.停止两个节点集群 停止之前最好ocrdump一下,防止有问题 crsctl stop crs 2.修改public ip地址和/etc/hosts 3. 启动crs 这时集群可以启动,但是上面的一些资源启动会…...
【Django-DRF用法】多年积累md笔记,第(4)篇:Django-DRF反序列化详解
本文从分析现在流行的前后端分离Web应用模式说起,然后介绍如何设计REST API,通过使用Django来实现一个REST API为例,明确后端开发REST API要做的最核心工作,然后介绍Django REST framework能帮助我们简化开发REST API的工作。 全…...
OpenAI宣布暂停ChatGPT plus用户订阅,解决方案,无需等待立马升级
作为人工智能领域的一项重要革新,ChatGPT Plus的上线引起了众多用户的关注,其背后的OpenAI表现出傲娇的态度,被誉为下一个GTP 4.0。总的来说,ChatGPT Plus的火爆主要有两个原因。首先,其在人工智能对话技术领域的创新&…...
如何将 Docsify 项目部署到 CentOS 系统的 Nginx 中
文章目录 第一步:准备 CentOS 服务器第二步:安装 Node.js 和 Docsify第三步:初始化 Docsify 项目第四步:本地预览 Docsify 项目第五步:配置 Nginx 服务器第六步:重启 Nginx 服务器拓展:使用 HTT…...
小程序存在优惠卷遍历,但是歪了
进入小程序,因为是一个小商城,所以照例先查看收货地址是否存在越权,以及能否未授权访问,但是发现不存在这些问题,所以去查看优惠卷 进入领券中心,点击领取优惠券时抓包 发现数据包,存在敏感参数…...
HarmonyOS第一课-对比Kotlin,快速入门TypeScript
编程语言简介 基础类型 1. 布尔值 TypeScript 和 Kotlin: 两者都有 boolean 类型,用于表示 true 或 false。 ts示例: let isDone:boolean falsekotlin示例: val isDone: Boolean false2. 数字 TypeScript: 有 number 类型,…...
【自动驾驶】一些业内自动驾驶专业术语释义
Trajectory 轨迹信息,一般都会发布未来5-10秒的trajactory信息。 Trajectory flicker 轨迹抖动 Nudge 道内避障。在维持车道不变的情况下,横向偏离车道中心以绕开obstacle/agent。 Xlane Nudge 借道避障。借用对向车道或自行车道以绕开obstacle/a…...
好用的博客评论系统 Valine 使用及避坑指南
评论系统,即网站的一个小功能,展示评论内容和用户输入框。开源免费的评论系统可不多,原来很火的"多说"评论系统都关闭了,而Disqus又是国外的访问受限。无意间发现了Valine,挺不错的,分享给大家。…...
【Mysql】[Err] 1293 - Incorrect table definition;
基本情况 SQL文件描述 /* Navicat MySQL Data TransferSource Server : cm4生产-200 Source Server Version : 50725 Source Host : 192.168.1.200:3306 Source Database : db_wmsTarget Server Type : MYSQL Target Server Version : 50725 File…...
Leetcode 3576. Transform Array to All Equal Elements
Leetcode 3576. Transform Array to All Equal Elements 1. 解题思路2. 代码实现 题目链接:3576. Transform Array to All Equal Elements 1. 解题思路 这一题思路上就是分别考察一下是否能将其转化为全1或者全-1数组即可。 至于每一种情况是否可以达到…...
大数据零基础学习day1之环境准备和大数据初步理解
学习大数据会使用到多台Linux服务器。 一、环境准备 1、VMware 基于VMware构建Linux虚拟机 是大数据从业者或者IT从业者的必备技能之一也是成本低廉的方案 所以VMware虚拟机方案是必须要学习的。 (1)设置网关 打开VMware虚拟机,点击编辑…...
Linux简单的操作
ls ls 查看当前目录 ll 查看详细内容 ls -a 查看所有的内容 ls --help 查看方法文档 pwd pwd 查看当前路径 cd cd 转路径 cd .. 转上一级路径 cd 名 转换路径 …...
2021-03-15 iview一些问题
1.iview 在使用tree组件时,发现没有set类的方法,只有get,那么要改变tree值,只能遍历treeData,递归修改treeData的checked,发现无法更改,原因在于check模式下,子元素的勾选状态跟父节…...
ESP32 I2S音频总线学习笔记(四): INMP441采集音频并实时播放
简介 前面两期文章我们介绍了I2S的读取和写入,一个是通过INMP441麦克风模块采集音频,一个是通过PCM5102A模块播放音频,那如果我们将两者结合起来,将麦克风采集到的音频通过PCM5102A播放,是不是就可以做一个扩音器了呢…...
【配置 YOLOX 用于按目录分类的图片数据集】
现在的图标点选越来越多,如何一步解决,采用 YOLOX 目标检测模式则可以轻松解决 要在 YOLOX 中使用按目录分类的图片数据集(每个目录代表一个类别,目录下是该类别的所有图片),你需要进行以下配置步骤&#x…...
OpenPrompt 和直接对提示词的嵌入向量进行训练有什么区别
OpenPrompt 和直接对提示词的嵌入向量进行训练有什么区别 直接训练提示词嵌入向量的核心区别 您提到的代码: prompt_embedding = initial_embedding.clone().requires_grad_(True) optimizer = torch.optim.Adam([prompt_embedding...
Java + Spring Boot + Mybatis 实现批量插入
在 Java 中使用 Spring Boot 和 MyBatis 实现批量插入可以通过以下步骤完成。这里提供两种常用方法:使用 MyBatis 的 <foreach> 标签和批处理模式(ExecutorType.BATCH)。 方法一:使用 XML 的 <foreach> 标签ÿ…...
pycharm 设置环境出错
pycharm 设置环境出错 pycharm 新建项目,设置虚拟环境,出错 pycharm 出错 Cannot open Local Failed to start [powershell.exe, -NoExit, -ExecutionPolicy, Bypass, -File, C:\Program Files\JetBrains\PyCharm 2024.1.3\plugins\terminal\shell-int…...
API网关Kong的鉴权与限流:高并发场景下的核心实践
🔥「炎码工坊」技术弹药已装填! 点击关注 → 解锁工业级干货【工具实测|项目避坑|源码燃烧指南】 引言 在微服务架构中,API网关承担着流量调度、安全防护和协议转换的核心职责。作为云原生时代的代表性网关,Kong凭借其插件化架构…...
