开发环境配置

安装GCC,GDB

  1. sudo apt update
  2. # 通过以下命令安装编译器和调试器
  3. sudo apt install build-essential gdb
  • 安装成功确认
  1. # 以下命令确认每个软件是否安装成功
  2. # 如果成功,则显示版本号
  3. gcc --version
  4. g++ --version
  5. gdb --version

CMake安装

  1. # 通过以下命令安装编译器和调试器
  2. sudo apt install cmake
  3. # 确认是否安装成功
  4. # 如果成功,则显示版本号
  5. cmake --version

gcc编译器

预热:

  1. GCC 编译器支持编译 Go、Objective-C,Objective-C ++,Fortran,Ada,D 和 BRIG(HSAIL)等程序;
  2. VSCode是通过调用GCC编译器来实现C/C++的编译工作的;
  • 实际使用中

    • 使用 gcc 指令编译 C 代码
    • 使用 g++指令编译 C++ 代码

编译过程

  1. 1. 预处理-Pre-Processing //.i文件
  2. # -E 选项指示编译器仅对输入文件进行预处理
  3. g++ -E test.cpp -o test.i //.i文件
  4. 2. 编译-Compiling (变成汇编语言) // .s文件
  5. # -S 编译选项告诉 g++ 在为 C++ 代码产生了汇编语言文件后停止编译
  6. # g++ 产生的汇编语言文件的缺省扩展名是 .s
  7. g++ -S test.i -o test.s
  8. 3. 汇编-Assembling (变成二进制语言) // .o文件
  9. # -c 选项告诉 g++ 仅把源代码编译为机器语言的目标代码
  10. # 缺省时 g++ 建立的目标代码文件有一个 .o 的扩展名。
  11. g++ -c test.s -o test.o
  12. 4. 链接-Linking // bin文件
  13. # -o 编译选项来为将产生的可执行文件用指定的文件名
  14. g++ test.o -o test

g++重要编译参数

  1. -g 编译带调试信息的可执行文件
  1. # -g 选项告诉 GCC 产生能被 GNU 调试器GDB使用的调试信息,以调试程序。
  2. # 产生带调试信息的可执行文件test
  3. g++ -g test.cpp
  1. -O[n] 优化源代码
  1. ## 所谓优化,例如省略掉代码中从未使用过的变量、直接将常量表达式用结果值代替等等,这些操作会缩减目标文件所包含的代码量,提高最终生成的可执行文件的运行效率。
  2. # -O 选项告诉 g++ 对源代码进行基本优化。这些优化在大多数情况下都会使程序执行的更快。 -O2 选项告诉 g++ 产生尽可能小和尽可能快的代码。 如-O2,-O3,-On(n 常为0–3)
  3. # -O 同时减小代码的长度和执行时间,其效果等价于-O1
  4. # -O0 表示不做优化
  5. # -O1 为默认优化
  6. # -O2 除了完成-O1的优化之外,还进行一些额外的调整工作,如指令调整等。
  7. # -O3 则包括循环展开和其他一些与处理特性相关的优化工作。
  8. # 选项将使编译的速度比使用 -O 时慢, 但通常产生的代码执行速度会更快。
  9. # 使用 -O2优化源代码,并输出可执行文件
  10. g++ -O2 test.cpp
  1. -l(库文件名字) 和 -L(库文件路径) 指定库文件 | 指定库文件路径
  1. # -l参数(小写)就是用来指定程序要链接的库,-l参数紧接着就是库名
  2. # 在/lib和/usr/lib和/usr/local/lib里的库直接用-l参数就能链接
  3. # 链接glog库
  4. g++ -lglog test.cpp
  5. # 如果库文件没放在上面三个目录里,需要使用-L参数(大写)指定库文件所在目录
  6. # -L参数跟着的是库文件所在的目录名
  7. # 链接mytest库,libmytest.so在/home/bing/mytestlibfolder目录下
  8. g++ -L/home/bing/mytestlibfolder -lmytest test.cpp
  1. -I 指定头文件搜索目录
  1. # -I
  2. # /usr/include目录一般是不用指定的,gcc知道去那里找,但 是如果头文件不在/usr/icnclude里我们就要用-I参数指定了,比如头文件放在/myinclude目录里,那编译命令行就要加上-I/myinclude 参数了,如果不加你会得到一个”xxxx.h: No such file or directory”的错误。-I参数可以用相对路径,比如头文件在当前 目录,可以用-I.来指定。上面我们提到的–cflags参数就是用来生成-I参数的。
  3. g++ -I/myinclude test.cpp
  1. -Wall 打印警告信息
  1. # 打印出gcc提供的警告信息
  2. g++ -Wall test.cpp
  1. -w 关闭警告信息
  1. # 关闭所有警告信息
  2. g++ -w test.cpp
  1. -std=c++11 设置编译标准
  1. # 使用 c++11 标准编译 test.cpp
  2. g++ -std=c++11 test.cpp
  1. -o 指定输出文件名
  1. # 指定即将产生的文件名
  2. # 指定输出可执行文件名为test
  3. g++ test.cpp -o test
  1. -D 定义宏
  1. # 在使用gcc/g++编译的时候定义宏
  2. # 常用场景:
  3. # -DDEBUG 定义DEBUG宏,可能文件中有DEBUG宏部分的相关信息,用个DDEBUG来选择开启或关闭DEBUG
  • 实例代码
  1. 1 // -Dname 定义宏name,默认定义内容为字符串“1”
  2. 2
  3. 3 #include <stdio.h>
  4. 4
  5. 5 int main()
  6. 6 {
  7. 7 #ifdef DEBUG
  8. 8 printf("DEBUG LOG\n");
  9. 9 #endif
  10. 10 printf("in\n");
  11. 11 }
  12. 12
  13. 13 // 1. 在编译的时候,使用gcc -DDEBUG main.cpp
  14. 14 // 2. 第七行代码可以被执行

实战: g++命令行编译

  • 案例:最初目录结构: 2 directories, 3 files
  1. # 最初目录结构
  2. .
  3. ├── include
  4. └── Swap.h
  5. ├── main.cpp
  6. └── src
  7. └── Swap.cpp
  8. 2 directories, 3 files
  1. 直接编译
  • 最简单的编译,并运行
  1. # 将 main.cpp src/Swap.cpp 编译为可执行文件
  2. g++ main.cpp src/Swap.cpp -Iinclude
  3. # 运行a.out
  4. ./a.out
  • 增加参数编译,并运行
  1. # 将 main.cpp src/Swap.cpp 编译为可执行文件 附带一堆参数
  2. g++ main.cpp src/Swap.cpp -Iinclude -std=c++11 -O2 -Wall -o b.out
  3. # 运行 b.out
  4. ./b.out

2. 生成库文件并编译

  • 链接静态库生成可执行文件①:
  1. ## 进入src目录下
  2. $cd src
  3. # 汇编,生成Swap.o文件
  4. g++ Swap.cpp -c -I../include
  5. # 生成静态库libSwap.a
  6. ar rs libSwap.a Swap.o
  7. ## 回到上级目录
  8. $cd ..
  9. # 链接,生成可执行文件:staticmain
  10. g++ main.cpp -Iinclude -Lsrc -lSwap -o staticmain
  11. 静态库文件通常以.a结尾
  12. 二进制文件通常以.o结尾

  • 链接动态库生成可执行文件②:
  1. ## 进入src目录下
  2. $cd src
  3. # 生成动态库libSwap.so
  4. g++ Swap.cpp -I../include -fPIC -shared -o libSwap.so
  5. ## 上面命令等价于以下两条命令
  6. # gcc Swap.cpp -I../include -c -fPIC
  7. # gcc -shared -o libSwap.so Swap.o
  8. ## 回到上级目录
  9. $cd ..
  10. # 链接,生成可执行文件:sharemain
  11. g++ main.cpp -Iinclude -Lsrc -lSwap -o sharemain
  12. -FPIC与路径无关的选项
  13. -shared生成动态库文件
  14. 动态库文件的扩展名.so
  15. -lswap 指定库文件名
  16. -Lsrc 库文件名路径
  • 目录结构
  1. # 最终目录结构
  2. .
  3. ├── include
  4. └── Swap.h
  5. ├── main.cpp
  6. ├── sharemain
  7. ├── src
  8. ├── libSwap.a
  9. ├── libSwap.so
  10. ├── Swap.cpp
  11. └── Swap.o
  12. └── staticmain
  13. 142 directories, 8 files

运行可执行文件

  1. 运行可执行文件①
  2. # 运行可执行文件 --静态库
  3. ./staticmain
  4. 运行可执行文件② -- 动态库
  5. # 运行可执行文件
  6. LD_LIBRARY_PATH=src ./sharemain

CMake

语法特性介绍

  • 基本语法格式: 指令(参数 1 参数 2…)
  1. 参数使用括弧括起
  2. 参数之间使用空格或分号分开
  • 指令是大小写无关的,参数和变量是大小写相关的
  1. set(HELLO hello.cpp)
  2. add_executable(hello main.cpp hello.cpp)
  3. ADD_EXECUTABLE(hello main.cpp ${HELLO})
  • 变量使用${}方式取值,但是在 IF 控制语句中是直接使用变量名

重要指令和CMake常用变量

重要指令

  • cmake_minimum_required - 指定CMake的最小版本要求
  1. # CMake最小版本要求为2.8.3
  2. cmake_minimum_required(VERSION 2.8.3)
  3. 语法:cmake_minimum_required(VERSION versionNumber [FATAL_ERROR])
  • project - 定义工程名称,并可指定工程支持的语言
  1. # 指定工程名为HELLOWORLD
  2. project(HELLOWORLD)
  3. 语法:project(projectname [CXX] [C] [Java])-->[]为可选
  • set - 显式的定义变量
  1. # 定义SRC变量,其值为main.cpp hello.cpp
  2. set(SRC sayhello.cpp hello.cpp)
  3. 语法:set(VAR [VALUE] [CACHE TYPE DOCSTRING [FORCE]])
  • include_directories - 向工程添加多个特定的头文件搜索路径 --->相当于指定g++编译器的-I参数
  1. # 将/usr/include/myincludefolder 和 ./include 添加到头文件搜索路径
  2. include_directories(/usr/include/myincludefolder ./include)
  3. 语法:include_directories([AFTER|BEFORE] [SYSTEM] dir1 dir2 …)
  • link_directories - 向工程添加多个特定的库文件搜索路径 --->相当于指定g++编译器的-L参数
  1. # 将/usr/lib/mylibfolder 和 ./lib 添加到库文件搜索路径
  2. link_directories(/usr/lib/mylibfolder ./lib)
  3. 语法:link_directories(dir1 dir2 …)
  • add_library - 生成库文件
  1. # 通过变量 SRC 生成 libhello.so 共享库
  2. add_library(hello SHARED ${SRC})
  3. 语法:add_library(libname [SHARED|STATIC|MODULE] [EXCLUDE_FROM_ALL] source1 source2 sourceN)
  • add_compile_options - 添加编译参数
  1. # 添加编译参数 -Wall -std=c++11
  2. add_compile_options(-Wall -std=c++11 -O2)
  • add_executable - 生成可执行文件
  1. # 编译main.cpp生成可执行文件main
  2. add_executable(main main.cpp)
  3. 语法:add_library(exename source1 source2 sourceN)
  • target_link_libraries - 为 target 添加需要链接的共享库 --->相同于指定g++编译器-l参数
  1. # 将hello动态库文件链接到可执行文件main
  2. target_link_libraries(main hello)
  3. 语法:target_link_libraries(target library1library2…)
  • add_subdirectory - 向当前工程添加存放源文件的子目录,并可以指定中间二进制和目标二进制存放的位置
  1. # 添加src子目录,src中需有一个CMakeLists.txt
  2. add_subdirectory(src)
  3. 语法:add_subdirectory(source_dir [binary_dir] [EXCLUDE_FROM_ALL])
  • aux_source_directory - 发现一个目录下所有的源代码文件并将列表存储在一个变量中,这个指令临时被用来自动构建源文件列表
  1. # 定义SRC变量,其值为当前目录下所有的源代码文件
  2. aux_source_directory(. SRC)
  3. # 编译SRC变量所代表的源代码文件,生成main可执行文件
  4. add_executable(main ${SRC})
  5. 语法:aux_source_directory(dir VARIABLE)

两种构建方式

内部构建(in-source build):不推荐使用

  1. ## 内部构建
  2. # 在当前目录下,编译本目录的CMakeLists.txt,生成Makefile和其他文件
  3. cmake .
  4. # 执行make命令,生成target
  5. make

外部构建(out-of-source build):推荐使用

  1. ## 外部构建
  2. # 1. 在当前目录下,创建build文件夹
  3. mkdir build
  4. # 2. 进入到build文件夹
  5. cd build
  6. # 3. 编译上级目录的CMakeLists.txt,生成Makefile和其他文件
  7. cmake ..
  8. # 4. 执行make命令,生成target
  9. make

CMake进行C/C++开发(linux下)的更多相关文章

  1. 嵌入式Linux应用开发——Linux下的C编程基础

    一.markdown简单操作 1.标题 在文字开头加上 “#”,通过“#”数量表示几级标题. 通过在文字下方添加“=”和“-”,他们分别表示一级标题和二级标题. 2.块注释 通过在文字开头添加“> ...

  2. [ARM-Linux开发]Linux下加载.ko驱动模块的两种方法:insmod与modprobe

    假设要加载的驱动程序模块名为SHT21.ko 加载驱动模块 方法一:  进入SHT21.ko驱动模块文件所在的目录,然后直接  insmod SHT21.ko  即可 方法二:  将SHT21.ko文 ...

  3. 大数据开发-linux下常见问题详解

    1.user ss is currently user by process 3234 问题原因:root --> ss --> root 栈递归一样 解决方式:exit 退出当前到ss再 ...

  4. linux下的C++项目创建

    CMake项目的完整构建 Linux下的CMake项目通常由几个文件夹组成.小伙伴们可以先在自己的电脑上新建一个文件夹,作为你代码的根目录,然后往里面建几个子文件夹,这里并不涉及具体的代码,只是可以作 ...

  5. linux下驱动webcam

    linux自带驱动只支持一些型号的camera,具体型号见http://www.ideasonboard.org/uvc/ 所以有些购买的webcam不能够在linux中被点亮,而且有些厂家只为了在W ...

  6. Linux下C++/C的编译调试

    这几天因为任务的原因我需要在ubuntu下编写程序.因此恶补了许多linux程序编写的知识.我分以下几个方面总结我所学的知识. gcc,g++,make命令的使用 gdb 调试 VScode的使用 c ...

  7. Linux下服务器端开发流程及相关工具介绍(C++)

    去年刚毕业来公司后,做为新人,发现很多东西都没有文档,各种工具和地址都是口口相传的,而且很多时候都是不知道有哪些工具可以使用,所以当时就想把自己接触到的这些东西记录下来,为后来者提供参考,相当于一个路 ...

  8. 在ubunt14.04(linux)下利用cmake编译运行opencv程序

    今天在电脑上安装好了opencv环境,迫不及待的想写个程序来测试一下.但是在windows下我们用vs等集成开发工具.可是在linux下我们应该怎么办呢? 这里我们用了opencv推荐的cmake来编 ...

  9. 【转载】Visual Studio 2015 for Linux更好地支持Linux下的开发

    原文:Visual Studio 2015 for Linux更好地支持Linux下的开发 英文原文:Targeting Linux Made Easier in Visual Studio 2015 ...

随机推荐

  1. call、apply和 bind的简单使用方法

    - call.apply:在 javascript 中,call 和 apply 都是为了改变某个函数运行时的上下文(context)而存在的,换句话说,就是为了改变函数体内部 this 的指向. J ...

  2. iOS开发 将html 富文本文字 转换成oc 的富文本

    - (NSMutableAttributedString *)mf_htmlAttribute:(NSString *)htmlString{ htmlString = [NSString strin ...

  3. redis5.0.0集群搭建【实战经历】

    redis集群搭建 作者:陈土锋 时间:2020年6月2日 目录 一.环境介绍... 1 1.机器准备... 1 2.关闭防护墙和selinux. 1 3.时间同步... 1 二.Redis Clus ...

  4. MySQL存储引擎、基础数据类型、约束条件

    MySQL存储引擎 存储引擎 # 存储引擎可以堪称是处理数据的不同方式 # 查看存储引擎的方式 show engines; # 需要掌握的四个存储引擎 MyISAM MySQL5.5之前的默认的存储引 ...

  5. Linux curl遇到错误curl: (3) Illegal characters found in URL

    服务器上执行一个脚本,在linux新建的sh,把本地编辑器的内容粘贴到文件里. 结果执行的时候报错了. 问题就是 curl:(3)Illegal characters found in URL 看着一 ...

  6. flex 布局 自己做的demo

    .row {/*横*/ display: -webkit-box;// display: -webkit-flex; /* Safari */ display: -moz-box; display: ...

  7. 2021.12.07 P4291 [HAOI2008]排名系统(Treap)

    2021.12.07 P4291 [HAOI2008]排名系统(Treap) https://www.luogu.com.cn/problem/P4291 双倍经验: https://www.luog ...

  8. JavaScript函数中的arguments对象

    ECMAScript标准中,每个函数都有一个特殊的内置对象arguments.arguments对象是一个类Array对象(object),用以保存函数接收到的实参副本. 一.内置特性 说它是一个内置 ...

  9. 论文解读(DCRN)《Deep Graph Clustering via Dual Correlation Reduction》

    论文信息 论文标题:Deep Graph Clustering via Dual Correlation Reduction论文作者:Yue Liu, Wenxuan Tu, Sihang Zhou, ...

  10. Java基础语法Day_07(1-3 常用API第一部分)

    常用API第一部分 第1节 Scanner类         day07_01_API概述和使用步骤(使用最基本的三个步骤 搜索 构造方法  方法) day07_02_Scanner概述及其API文档 ...