本次课程适合的对象:
Linux零基础,想了解最常用的高频Linux命令的小伙伴
只会在Windows开发C/C++,需要转Linux开发的小伙伴
想深入了解并掌握GCC编译器编译语法和规则的小伙伴
想深入了解并掌握GDB调试器命令行调试的小伙伴
想学习使用CMake构建C/C++工程的小伙伴
想学习Linux下使用VSCode进行C/C++开发的小伙伴
总而言之,本课程将从零开始,教会你如何在Linux开发C/C++,带领你一起打开新世界的大门~
系统环境:Ubuntu18.04 LTS 虚拟机
开发语言:C++
开发IDE :VSCode
课程宗旨:无干货,不视频。分享让生活更美好。
什么是linux系统?
Linux是开源的操作系统
多用户多任务:
单用户:一个用户,在登录计算机(操作系统),只能允许同时登录一个用户;
单任务:一个任务,允许用户同时进行的操作任务数量;
多用户:多个用户,在登录计算机(操作系统),允许同时登录多个用户进行操作;
多任务:多个任务,允许用户同时进行多个操作任务;
Windows属于:单用户、多任务。
Linux属于:多用户、多任务。
Linux一切皆文件
对于文件的操作的都有哪些种类?
创建文件、编辑文件、保存文件、关闭文件、重命名文件、删除文件、恢复文件。
目录结构:
Bin:全称binary,含义是二进制。该目录中存储的都是一些二进制文件,文件都是可以被运行的。
Dev:该目录中主要存放的是外接设备,例如盘、其他的光盘等。在其中的外接设备是不能直接被使用的,需要挂载(类似window下的分配盘符)。
Etc:该目录主要存储一些配置文件。
Home:*表示“家”,表示*除了root用户以外其他用户的家目录,类似于windows下的User/用户目录。
Proc:全称process,表示进程,该目录中存储的是Linux运行时候的进程。
Root:该目录是root用户自己的家目录。
Sbin:全称super binary,该目录也是存储一些可以被执行的二进制文件,但是必须得有super权限的用户才能执行。
Tmp:表示“临时”的,当系统运行时候产生的临时文件会在这个目录存着。
Usr:存放的是用户自己安装的软件。类似于windows下的program files。
Var:存放的程序/系统的日志文件的目录。
Mnt:当外接设备需要挂载的时候,就需要挂载到mnt目录下。
指令含义:
Linux的指令是指在Linux终端(命令行)中输入的内容。
指令格式:
完整指令的标准格式:命令(空格) [选项](空格) [操作对象]
选项和操作对象都可以没有,也可以是多个
xxxxxxxxxx31# 指令示例:以下两条指令等价2ls -l -a -h /home ./3ls -lah /home ./pwd - Print current working directory
作用:打印当前终端所在的目录
用法: pwd
xxxxxxxxxx21# 打印当前目录2pwdls - List directory contents
作用:列出当前工作目录下的所有文件/文件夹的名称
用法1:ls
含义:列出当前工作目录下的文件/文件夹的名称
xxxxxxxxxx11ls用法2:ls [路径]
含义:*列出指定*路径下的所有文件/文件夹的名称
绝对路径:相对根目录的路径;
相对路径:相对当前目录的路径;
xxxxxxxxxx51# ls 相对路径2ls ./ #【表示当前目录下】3ls ../ #【上一级目录下】4# ls 绝对路径5ls /home用法3:ls [选项] [路径]
含义:在列出指定路径下的文件/文件夹的名称,并以指定的格式进行显示。
xxxxxxxxxx81# ls 选项 路径2ls -lah /home3# 选项解释:4 -l:表示list,表示以详细列表的形式进行展示5 -a:表示显示所有的文件/文件夹(包含了隐藏文件/文件夹)6 -h:表示以可读性较高的形式显示7# ls -l 中 “-”表示改行对应的文档类型为文件,“d”表示文档类型为文件夹。8# 在Linux中隐藏文档一般都是以“.”开头cd - change directory
作用:切换当前的工作目录
用法1:cd ; cd ~
xxxxxxxxxx31# 以下两条命令等价,示直接进入当前用户的家目录下【很常用】2cd3cd ~用法2:cd [相对路径]
xxxxxxxxxx41# 进入到上级目录下2cd ..3# 进入到上级目录中的local目录下4cd ../local用法3:cd [绝对路径]
xxxxxxxxxx21# 进入到/usr/local目录下2cd /usr/localmkdir - make directories
作用:创建目录
用法1:mkdir 路径
xxxxxxxxxx21# 在当前路径下创建出目录“myfolder”2mkdir myfolder用法2:mkdir -p 路径
含义:一次性创建多层不存在的目录
xxxxxxxxxx21# 创建 ~/a/b/c 目录2mkdir -p ~/a/b/c用法3:mkdir 路径1 [路径2] [路径3]
含义:一次性创建多个目录
xxxxxxxxxx21# 在当前目录分别创建a、b、c三个文件夹2mkdir a b ctouch - change file timestamps
作用:创建新文件
xxxxxxxxxx51--------------------------------------------------------------------------2# 【为什么创建新文件是touch】3# 1. touch的作用本来不是创建文件,而是将指定文件的修改时间设置为当前时间。就是假装“碰”(touch)了一下这个文件,假装文件被“修改”了,于是文件的修改时间就是被设置为当前时间。4# 2. 这带来了一个副作用,就是当touch一个不存在的文件的时候,它会创建这个文件。然后,由于touch已经可以完成创建文件的功能了,就不再需要一个单独的create了。5--------------------------------------------------------------------------用法1:touch [路径]
x1# 在当前目录下创建linux.txt文件2touch linux.txt3
4# 在上级目录下创建linux文件5touch ../linux6
7# 在/home/bing/目录下创建myfile文件8touch /home/bing/myfile用法2:touch [路径1] [路径2]
xxxxxxxxxx21# 在当前目录下创建file file.txt 两个文件2touch file file.txtrm - remove files or directories
作用:删除文件/目录
用法1:rm [选项] 需要移除的文件路径
xxxxxxxxxx41# 删除当前路径下的myfile文件2rm myfile3# 删除/usr路径下的myfile文件4rm /usr/myfile用法2:rm [选项] 需要移除的目录
xxxxxxxxxx41# 删除当前路径下的abc文件2rm -rf myfolder3# 删除/usr路径下的abc文件4rm -rf /usr/myfoldercp - copy files and directories
作用:复制文件/文件夹到指定的位置
用法1:cp [被复制的文件路径] [文件被复制到的路径]
xxxxxxxxxx21# cp命令来复制一个文件2cp /home/bing/myfile ./用法2:cp -r [被复制的文件夹路径] [文件夹被复制到的路径]*
含义:-r 表示递归复制,复制文件夹的时候需要加 -r
xxxxxxxxxx21# 复制/home/bing/myfolder文件夹到根目录/下2cp -r /home/bing/myfolder /用法3:mkdir [路径1] [路径2] [路径3]
含义:一次性创建多个目录
mv - move (rename) files
作用:移动文件到新的位置,或者重命名文件
用法:mv [需要移动的文件路径] [需要保存的位置路径]
xxxxxxxxxx81# 移动当前目录下myfile文件到根目录/下2mv myfile /myfile3
4# 移动当前目录下myfolder文件夹到根目录/下5mv myfolder /myfolder6
7# 移动当前目录下myfile到根目录/下,并重命名为myfile0078mv myfile myfile007man - an interface to the system reference manuals
作用:包含了Linux中全部命令手册
用法:man [命令]
含义:查看命令使用手册,按 q 退出
xxxxxxxxxx61# 查看ls命令的手册2man ls3# 查看cd命令的手册4man cd5# 查看man命令的手册6man manreboot - reboot the machine
作用:重启linux系统
用法:reboot
xxxxxxxxxx21# 立即重启2rebootshutdown - power-off the machine
作用:关机
用法:shut -h [时间]
xxxxxxxxxx21# 立即关机2shutdown -h nowVim [file]
所有的 Linux系统都会内建 Vi/Vim编辑器,其他的编辑器则不一定会存在
Vim是所有Unix及Linux系统下标准的编辑器
Vim 具有程序开发的能力,也可以用来对文件进行简单的编辑
Vim具有“编辑器之神”的称号,学会Vim便可在Linux的世界里畅行无阻,尤其是在终端中。
👇Vim操作终极图片👇
gedit [file]
Linux 下的一个纯文本编辑器
可以根据不同的语言高亮显现关键字和标识符。
nano [file]
nano 是一个小巧的文本编辑器
它比vi/vim要简单得多,比较适合Linux初学者使用。
某些Linux发行版的默认编辑器就是nano。
安装GCC,GDB
xxxxxxxxxx31sudo apt update2# 通过以下命令安装编译器和调试器3sudo apt install build-essential gdb安装成功确认
xxxxxxxxxx51# 以下命令确认每个软件是否安装成功2# 如果成功,则显示版本号3gcc --version4g++ --version5gdb --version安装cmake
xxxxxxxxxx21# 通过以下命令安装编译器和调试器2sudo apt install cmake安装成功确认
xxxxxxxxxx31# 确认是否安装成功2# 如果成功,则显示版本号3cmake --version前言:
GCC 编译器支持编译 Go、Objective-C,Objective-C ++,Fortran,Ada,D 和 BRIG(HSAIL)等程序;
Linux 开发C/C++ 一定要熟悉 GCC
VSCode是通过调用GCC编译器来实现C/C++的编译工作的;
实际使用中:
使用 gcc 指令编译 C 代码
使用 g++指令编译 C++ 代码
预处理-Pre-Processing //.i文件
xxxxxxxxxx21# -E 选项指示编译器仅对输入文件进行预处理2g++ -E test.cpp -o test.i //.i文件编译-Compiling // .s文件
xxxxxxxxxx31# -S 编译选项告诉 g++ 在为 C++ 代码产生了汇编语言文件后停止编译2# g++ 产生的汇编语言文件的缺省扩展名是 .s3g++ -S test.i -o test.s汇编-Assembling // .o文件
xxxxxxxxxx31# -c 选项告诉 g++ 仅把源代码编译为机器语言的目标代码2# 缺省时 g++ 建立的目标代码文件有一个 .o 的扩展名。3g++ -c test.s -o test.o链接-Linking // bin文件
xxxxxxxxxx21# -o 编译选项来为将产生的可执行文件用指定的文件名2g++ test.o -o test-g 编译带调试信息的可执行文件
xxxxxxxxxx41# -g 选项告诉 GCC 产生能被 GNU 调试器GDB使用的调试信息,以调试程序。2
3# 产生带调试信息的可执行文件test4g++ -g test.cpp-O[n] 优化源代码
xxxxxxxxxx121## 所谓优化,例如省略掉代码中从未使用过的变量、直接将常量表达式用结果值代替等等,这些操作会缩减目标文件所包含的代码量,提高最终生成的可执行文件的运行效率。2
3# -O 选项告诉 g++ 对源代码进行基本优化。这些优化在大多数情况下都会使程序执行的更快。 -O2 选项告诉 g++ 产生尽可能小和尽可能快的代码。 如-O2,-O3,-On(n 常为0–3)4# -O 同时减小代码的长度和执行时间,其效果等价于-O15# -O0 表示不做优化6# -O1 为默认优化7# -O2 除了完成-O1的优化之外,还进行一些额外的调整工作,如指令调整等。8# -O3 则包括循环展开和其他一些与处理特性相关的优化工作。9# 选项将使编译的速度比使用 -O 时慢, 但通常产生的代码执行速度会更快。10
11# 使用 -O2优化源代码,并输出可执行文件12g++ -O2 test.cpp-l 和 -L 指定库文件 | 指定库文件路径
xxxxxxxxxx111# -l参数(小写)就是用来指定程序要链接的库,-l参数紧接着就是库名2# 在/lib和/usr/lib和/usr/local/lib里的库直接用-l参数就能链接3
4# 链接glog库5g++ -lglog test.cpp6
7# 如果库文件没放在上面三个目录里,需要使用-L参数(大写)指定库文件所在目录8# -L参数跟着的是库文件所在的目录名9
10# 链接mytest库,libmytest.so在/home/bing/mytestlibfolder目录下11g++ -L/home/bing/mytestlibfolder -lmytest test.cpp-I 指定头文件搜索目录
xxxxxxxxxx41# -I2# /usr/include目录一般是不用指定的,gcc知道去那里找,但 是如果头文件不在/usr/icnclude里我们就要用-I参数指定了,比如头文件放在/myinclude目录里,那编译命令行就要加上-I/myinclude 参数了,如果不加你会得到一个”xxxx.h: No such file or directory”的错误。-I参数可以用相对路径,比如头文件在当前 目录,可以用-I.来指定。上面我们提到的–cflags参数就是用来生成-I参数的。3
4g++ -I/myinclude test.cppWall 打印警告信息
xxxxxxxxxx21# 打印出gcc提供的警告信息2g++ -Wall test.cpp-w 关闭警告信息
xxxxxxxxxx21# 关闭所有警告信息2g++ -w test.cpp-std=c++11 设置编译标准
xxxxxxxxxx21# 使用 c++11 标准编译 test.cpp2g++ -std=c++11 test.cpp-o 指定输出文件名
xxxxxxxxxx41# 指定即将产生的文件名2
3# 指定输出可执行文件名为test4g++ test.cpp -o test-D 定义宏
xxxxxxxxxx41# 在使用gcc/g++编译的时候定义宏2
3# 常用场景:4# -DDEBUG 定义DEBUG宏,可能文件中有DEBUG宏部分的相关信息,用个DDEBUG来选择开启或关闭DEBUG示例代码:
xxxxxxxxxx141// -Dname 定义宏name,默认定义内容为字符串“1”2
34
5int main()6{7 8 printf("DEBUG LOG\\n");9 10 printf("in\\n");11}12
13// 1. 在编译的时候,使用gcc -DDEBUG main.cpp14// 2. 第七行代码可以被执行注:使用 man gcc命令可以查看gcc英文使用手册,见下图
案例:最初目录结构: 2 directories, 3 files
xxxxxxxxxx91# 最初目录结构2.3├── include4│ └── Swap.h 5├── main.cpp6└── src7 └── Swap.cpp8
92 directories, 3 files最简单的编译,并运行
xxxxxxxxxx41# 将 main.cpp src/Swap.cpp 编译为可执行文件2g++ main.cpp src/Swap.cpp -Iinclude3# 运行a.out4./a.out增加参数编译,并运行
xxxxxxxxxx41# 将 main.cpp src/Swap.cpp 编译为可执行文件 附带一堆参数2g++ main.cpp src/Swap.cpp -Iinclude -std=c++11 -O2 -Wall -o b.out3# 运行 b.out4./b.out链接静态库生成可执行文件①:
xxxxxxxxxx131## 进入src目录下2$cd src3
4# 汇编,生成Swap.o文件5g++ Swap.cpp -c -I../include6# 生成静态库libSwap.a7ar rs libSwap.a Swap.o8
9## 回到上级目录10$cd ..11
12# 链接,生成可执行文件:staticmain13g++ main.cpp -Iinclude -Lsrc -lSwap -o staticmain链接动态库生成可执行文件②:
xxxxxxxxxx141## 进入src目录下2$cd src3
4# 生成动态库libSwap.so5g++ Swap.cpp -I../include -fPIC -shared -o libSwap.so6## 上面命令等价于以下两条命令7# gcc Swap.cpp -I../include -c -fPIC8# gcc -shared -o libSwap.so Swap.o9
10## 回到上级目录11$cd ..12
13# 链接,生成可执行文件:sharemain14g++ main.cpp -Iinclude -Lsrc -lSwap -o sharemain编译完成后的目录结构
最终目录结构:2 directories, 8 files
xxxxxxxxxx151# 最终目录结构2.3├── include4│ └── Swap.h5
6├── main.cpp7├── sharemain8├── src9│ ├── libSwap.a10│ ├── libSwap.so11│ ├── Swap.cpp12│ └── Swap.o13└── staticmain14
152 directories, 8 files运行可执行文件①
xxxxxxxxxx21# 运行可执行文件2./staticmain
运行可执行文件②
xxxxxxxxxx21# 运行可执行文件2LD_LIBRARY_PATH=src ./sharemain前言:
GDB(GNU Debugger)*是一个用来*调试C/C++程序的功能强大的调试器,是Linux系统开发C/C++最常用的调试器
程序员可以使用GDB来跟踪程序中的错误,从而减少程序员的工作量。
Linux 开发C/C++ 一定要熟悉 GDB
VSCode是通过调用GDB调试器来实现C/C++的调试工作的;
Windows 系统中,常见的集成开发环境(IDE),如 VS、VC等,它们内部已经嵌套了相应的调试器
GDB主要功能:
设置断点(断点可以是条件表达式)
使程序在指定的代码行上暂停执行,便于观察
单步执行程序,便于调试
查看程序中变量值的变化
动态改变程序的执行环境
分析崩溃程序产生的core文件
调试开始:执行gdb [exefilename] ,进入gdb调试程序,其中exefilename为要调试的可执行文件名
xxxxxxxxxx521## 以下命令后括号内为命令的简化使用,比如run(r),直接输入命令 r 就代表命令run2
3$(gdb)help(h) # 查看命令帮助,具体命令查询在gdb中输入help + 命令4
5$(gdb)run(r) # 重新开始运行文件(run-text:加载文本文件,run-bin:加载二进制文件)6
7$(gdb)start # 单步执行,运行程序,停在第一行执行语句8
9$(gdb)list(l) # 查看原代码(list-n,从第n行开始查看代码。list+ 函数名:查看具体函数)10
11$(gdb)set # 设置变量的值12
13$(gdb)next(n) # 单步调试(逐过程,函数直接执行)14
15$(gdb)step(s) # 单步调试(逐语句:跳入自定义函数内部执行)16
17$(gdb)backtrace(bt) # 查看函数的调用的栈帧和层级关系18
19$(gdb)frame(f) # 切换函数的栈帧20
21$(gdb)info(i) # 查看函数内部局部变量的数值22
23$(gdb)finish # 结束当前函数,返回到函数调用点24
25$(gdb)continue(c) # 继续运行26
27$(gdb)print(p) # 打印值及地址28
29$(gdb)quit(q) # 退出gdb30$(gdb)break+num(b) # 在第num行设置断点31
32$(gdb)info breakpoints # 查看当前设置的所有断点33
34$(gdb)delete breakpoints num(d) # 删除第num个断点35
36$(gdb)display # 追踪查看具体变量值37
38$(gdb)undisplay # 取消追踪观察变量39
40$(gdb)watch # 被设置观察点的变量发生修改时,打印显示41
42$(gdb)i watch # 显示观察点43
44$(gdb)enable breakpoints # 启用断点45
46$(gdb)disable breakpoints # 禁用断点47
48$(gdb)x # 查看内存x/20xw 显示20个单元,16进制,4字节每单元49
50$(gdb)run argv[1] argv[2] # 调试时命令行传参51
52$(gdb)set follow-fork-mode child#Makefile项目管理:选择跟踪父子进程(fork())Tips:
编译程序时需要加上-g,之后才能用gdb进行调试:gcc -g main.c -o main
回车键:重复上一命令
给出一段简单代码,准备调试。
xxxxxxxxxx2112using namespace std;3
4int main(int argc,char **argv)5{6 int N = 100;7 int sum = 0;8 int i = 1;9
10// calculate sum from 1 to 10011 while (i <= N)12 {13 sum = sum + i;14 i = i + 1;15 }16
17 cout << "sum = " << sum << endl;18 cout << "The program is over." << endl;19
20 return 0;21}侧边栏
菜单栏
编辑区
状态栏
以下三款插件是在Linux下开发C/C++的三款必备插件~
C/C++
CMake
CMake Tools
高频使用快捷键:
| 功能 | 快捷键 | 功能 | 快捷键 |
|---|---|---|---|
| 转到文件 / 其他常用操作 | Ctrl + P | 关闭当前文件 | Ctrl + W |
| 打开命令面板 | Ctrl + Shift + P | 当前行上移/下移 | Alt + Up/Down |
| 打开终端 | Ctrl +`` | 变量统一重命名 |F2` | ||
| 关闭侧边栏 | Ctrl + B | 转到定义处 | F12 |
| 复制文本 | Ctrl+C | 粘贴文本 | Ctrl+V |
| 保存文件 | Ctrl+S | 撤销操作 | Ctrl+Z |
在 Ctrl+P 窗口下还可以:
直接输入文件名,跳转到文件
? 列出当前可执行的动作
! 显示 Errors或 Warnings,也可以 Ctrl+Shift+M
: 跳转到行数,也可以 Ctrl+G 直接进入
@ 跳转到 symbol(搜索变量或者函数),也可以 Ctrl+Shift+O 直接进入
@ 根据分类跳转 symbol,查找属性或函数,也可以 Ctrl+Shift+O 后输入:进入
# 根据名字查找 symbol,也可以 Ctrl+T
快捷键:编辑器与窗口管理
打开一个新窗口: Ctrl+Shift+N
关闭窗口: Ctrl+Shift+W
同时打开多个编辑器(查看多个文件)
新建文件 Ctrl+N
文件之间切换 Ctrl+Tab
切出一个新的编辑器(最多 3 个) Ctrl+\\,也可以按住 Ctrl 鼠标点击 Explorer 里的文件名
左中右 3 个编辑器的快捷键 Ctrl+1 Ctrl+2 Ctrl+3
3 个编辑器之间循环切换 Ctrl+
编辑器换位置, Ctrl+k然后按 Left或 Right
↓ 代码编辑相关的快捷键 ↓
快捷键:格式调整
代码行缩进 Ctrl+[ 、 Ctrl+]
Ctrl+C 、 Ctrl+V 复制或剪切当前行/当前选中内容
代码格式化: Shift+Alt+F,或 Ctrl+Shift+P 后输入 format code
上下移动一行: Alt+Up 或 Alt+Down
向上向下复制一行: Shift+Alt+Up 或 Shift+Alt+Down
在当前行下边插入一行 Ctrl+Enter
在当前行上方插入一行 Ctrl+Shift+Enter
快捷键:光标相关
移动到行首: Home
移动到行尾: End
移动到文件结尾: Ctrl+End
移动到文件开头: Ctrl+Home
移动到定义处: F12
定义处缩略图:只看一眼而不跳转过去 Alt+F12
移动到后半个括号: Ctrl+Shift+]
选择从光标到行尾: Shift+End
选择从行首到光标处: Shift+Home
删除光标右侧的所有字: Ctrl+Delete
扩展/缩小选取范围: Shift+Alt+Left 和 Shift+Alt+Right
多行编辑(列编辑):Alt+Shift+鼠标左键,Ctrl+Alt+Down/Up
同时选中所有匹配: Ctrl+Shift+L
Ctrl+D 下一个匹配的也被选中 (在 sublime 中是删除当前行,后面自定义快键键中,设置与 Ctrl+Shift+K 互换了)
回退上一个光标操作: Ctrl+U
快捷键:重构代码
找到所有的引用: Shift+F12
同时修改本文件中所有匹配的: Ctrl+F12
重命名:比如要修改一个方法名,可以选中后按 F2,输入新的名字,回车,会发现所有的文件都修改了
跳转到下一个 Error 或 Warning:当有多个错误时可以按 F8 逐个跳转
查看 diff: 在 explorer 里选择文件右键 Set file to compare,然后需要对比的文件上右键选择 Compare with file_name_you_chose
快捷键:查找替换
查找 Ctrl+F
查找替换 Ctrl+H
整个文件夹中查找 Ctrl+Shift+F
快捷键:显示相关
全屏:F11
zoomIn/zoomOut:Ctrl +/-
侧边栏显/隐:Ctrl+B
显示资源管理器 Ctrl+Shift+E
显示搜索 Ctrl+Shift+F
显示 Git Ctrl+Shift+G
显示 Debug Ctrl+Shift+D
显示 Output Ctrl+Shift+U
通过手写2个小项目,呈现出基本的C++工程建立的过程,并编译运行这两个小项目
项目1:Hello world
项目2:Swap with class
左右分屏
固定打开的文件
格式化代码
Rename Symbol
显示minimap
侧边栏查看OUTLINE
全屏
关闭多个打开的文件
前言:
CMake是一个跨平台的安装编译工具,可以用简单的语句来描述所有平台的安装(编译过程)。
CMake可以说已经成为大部分C++开源项目标配
Let’s assume you have some cross-platform project with C++ code shared along different platforms/IDEs. Say you use Visual Studio on Windows, Xcode on OSX and Makefile for Linux:
What you will do if you want to add new bar.cpp source file? You have to add it to every tool you use:
To keep the environment consistent you have to do the similar update several times. And the most important thing is that you have to do it manually (arrow marked with a red color on the diagram in this case). Of course such approach is error prone and not flexible.
CMake solve this design flaw by adding extra step to development process. You can describe your project in CMakeLists.txt file and use CMake to generate tools you currently interested in using cross-platform CMake code:
Same action - adding new bar.cpp file, will be done in one step now:
Note that the bottom part of the diagram was not changed. I.e. you still can keep using your favorite tools like Visual Studio/msbuild, Xcode/xcodebuild and Makefile/make!
基本语法格式:指令(参数 1 参数 2...)
参数使用括弧括起
参数之间使用空格或分号分开
指令是大小写无关的,参数和变量是大小写相关的
xxxxxxxxxx31set(HELLO hello.cpp)2add_executable(hello main.cpp hello.cpp)3ADD_EXECUTABLE(hello main.cpp ${HELLO})变量使用${}方式取值,但是在 IF 控制语句中是直接使用变量名
cmake_minimum_required 指定CMake的最小版本要求
语法: cmake_minimum_required(VERSION versionNumber [FATAL_ERROR])
xxxxxxxxxx21# CMake最小版本要求为2.8.32cmake_minimum_required(VERSION 2.8.3)project 定义工程名称,并可指定工程支持的语言
语法: project(projectname [CXX] [C] [Java])
xxxxxxxxxx21# 指定工程名为HELLOWORLD2project(HELLOWORLD)set 显式的定义变量
语法:set(VAR [VALUE] [CACHE TYPE DOCSTRING [FORCE]])
xxxxxxxxxx21# 定义SRC变量,其值为sayhello.cpp hello.cpp2set(SRC sayhello.cpp hello.cpp)include_directories 向工程添加多个特定的头文件搜索路径 --->相当于指定g++编译器的-I参数
语法: include_directories([AFTER|BEFORE] [SYSTEM] dir1 dir2 ...)
xxxxxxxxxx21# 将/usr/include/myincludefolder 和 ./include 添加到头文件搜索路径2include_directories(/usr/include/myincludefolder ./include)link_directories 向工程添加多个特定的库文件搜索路径 --->相当于指定g++编译器的-L参数
语法: link_directories(dir1 dir2 ...)
xxxxxxxxxx21# 将/usr/lib/mylibfolder 和 ./lib 添加到库文件搜索路径2link_directories(/usr/lib/mylibfolder ./lib)add_library 生成库文件
语法: add_library(libname [SHARED|STATIC|MODULE] [EXCLUDE_FROM_ALL] source1 source2 ... sourceN)
xxxxxxxxxx21# 通过变量 SRC 生成 libhello.so 共享库2add_library(hello SHARED ${SRC})add_compile_options 添加编译参数
语法:add_compile_options(
xxxxxxxxxx21# 添加编译参数 -Wall -std=c++11 -O22add_compile_options(-Wall -std=c++11 -O2)add_executable 生成可执行文件
语法:add_executable(exename source1 source2 ... sourceN)
xxxxxxxxxx21# 编译main.cpp生成可执行文件main2add_executable(main main.cpp)target_link_libraries 为 target 添加需要链接的共享库 -->相同于指定g++编译器-l参数
语法: target_link_libraries(target library1<debug | optimized> library2...)
xxxxxxxxxx21# 将hello动态库文件链接到可执行文件main2target_link_libraries(main hello)add_subdirectory 向当前工程添加存放源文件的子目录,并可以指定中间二进制和目标二进制存放的位置
语法: add_subdirectory(source_dir [binary_dir] [EXCLUDE_FROM_ALL])
xxxxxxxxxx21# 添加src子目录,src中需有一个CMakeLists.txt2add_subdirectory(src)aux_source_directory 发现一个目录下所有的源代码文件并将列表存储在一个变量中,这个指令临时被用来自动构建源文件列表
语法: aux_source_directory(dir VARIABLE)
xxxxxxxxxx41# 定义SRC变量,其值为当前目录下所有的源代码文件2aux_source_directory(. SRC)3# 编译SRC变量所代表的源代码文件,生成main可执行文件4add_executable(main ${SRC})CMAKE_C_FLAGS gcc编译选项
CMAKE_CXX_FLAGS g++编译选项
xxxxxxxxxx21# 在CMAKE_CXX_FLAGS编译选项后追加-std=c++112set( CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++11")CMAKE_BUILD_TYPE 编译类型(Debug, Release)
xxxxxxxxxx41# 设定编译类型为debug,调试时需要选择debug2set(CMAKE_BUILD_TYPE Debug)3# 设定编译类型为release,发布时需要选择release4set(CMAKE_BUILD_TYPE Release)CMAKE_BINARY_DIR
PROJECT_BINARY_DIR
这三个变量指代的内容是一致的。如果是 in source build,指的就是工程顶层目录。如果是 out-of-source 编译,指的是工程编译发生的目录。PROJECT_BINARY_DIR 跟其他指令稍有区别,不过现在,你可以理解为他们是一致的。
CMAKE_SOURCE_DIR
PROJECT_SOURCE_DIR
这三个变量指代的内容是一致的,不论采用何种编译方式,都是工程顶层目录。也就是在 in source build时,他跟 CMAKE_BINARY_DIR 等变量一致。PROJECT_SOURCE_DIR 跟其他指令稍有区别,现在,你可以理解为他们是一致的。
CMAKE_C_COMPILER:指定C编译器
CMAKE_CXX_COMPILER:指定C++编译器
EXECUTABLE_OUTPUT_PATH:可执行文件输出的存放路径
LIBRARY_OUTPUT_PATH:库文件输出的存放路径
CMake目录结构:项目主目录存在一个CMakeLists.txt文件
两种方式设置编译规则:
包含源文件的子文件夹包含CMakeLists.txt文件,主目录的CMakeLists.txt通过add_subdirectory添加子目录即可;
包含源文件的子文件夹未包含CMakeLists.txt文件,子目录编译规则体现在主目录的CMakeLists.txt中;
在 linux 平台下使用 CMake 构建C/C++工程的流程如下:
手动编写 CMakeLists.txt。
执行命令 cmake PATH生成 Makefile ( PATH 是顶层CMakeLists.txt 所在的目录 )。
执行命令make 进行编译。
xxxxxxxxxx61# important tips2. # 表示当前目录3./ # 表示当前目录4
5.. # 表示上级目录6../ # 表示上级目录内部构建(in-source build):不推荐使用
内部构建会在同级目录下产生一大堆中间文件,这些中间文件并不是我们最终所需要的,和工程源文件放在一起会显得杂乱无章。
xxxxxxxxxx61## 内部构建2
3# 在当前目录下,编译本目录的CMakeLists.txt,生成Makefile和其他文件4cmake .5# 执行make命令,生成target6make外部构建(out-of-source build):推荐使用
将编译输出文件与源文件放到不同目录中
xxxxxxxxxx101## 外部构建2
3# 1. 在当前目录下,创建build文件夹4mkdir build5# 2. 进入到build文件夹6cd build7# 3. 编译上级目录的CMakeLists.txt,生成Makefile和其他文件8cmake ..9# 4. 执行make命令,生成target10make针对第五章写的两个小项目来写对应的CMakeLists.txt
xxxxxxxxxx81# Set the minimum version of CMake that can be used2cmake_minimum_required(VERSION 3.0)3
4# Set the project name5project (HELLO)6
7# Add an executable8add_executable(hello_cmake main.cpp)xxxxxxxxxx171# Set the minimum version of CMake that can be used2cmake_minimum_required(VERSION 3.0)3
4#project name5project(SWAP)6
7#head file pat8include_directories( include )9
10#source directory files to var11add_subdirectory( src DIR_SRCS )12
13#add executable file14add_executable(swap_02 ${TEST_MATH})15
16#add link library17target_link_libraries(${FS_BUILD_BINARY_PREFIX}sqrt ${LIBRARIES})xxxxxxxxxx331# Set the minimum version of CMake that can be used2cmake_minimum_required(VERSION 3.0)3
4#project name5project(SWAP_LIBRARY)6
7#add compile options8add_compile_options("-Wall -std=c++11")9
10#set CMAKE_BUILD_TYPE11set( CMAKE_BUILD_TYPE Debug )12
13# set output binary path14set(EXECUTABLE_OUTPUT_PATH ${PROJECT_BINARY_DIR}/bin)15
16############################################################17# Create a library18############################################################19
20#Generate the static library from the library sources21add_library( swap_library STATIC src/Swap.cpp )22
23target_include_directories( swap_lib PUBLIC ${PROJECT_SOURCE_DIR}/include )24
25############################################################26# Create an executable27############################################################28
29# Add an executable with the above sources30add_executable( swap_01 main.cpp )31
32# link the new swap_01 target with the swap_lib target33target_link_libraries( swap_01 swap_liby )
士兵 许三多 有一把枪,叫做 AK47
士兵 可以 开火
士兵 可以 给枪装填子弹
枪 能够 发射 子弹
枪 能够 装填子弹 —— 增加子弹数量
开发枪类
开发士兵类