第三讲:GCC编译器
前沿:
1、GCC编译器支持编译Go、 Objective-C, Objective-C++, Fortran,Ada,D和BRIG( HSAIL)等程序;
2、Linux 开发C/C++ 一定要熟悉GCC;
3、VSCode是通过调用GCC编译器来实现C/C++的编译工作的。
实际使用过程中:
- 使用gcc指令编译C代码
- 使用g++指令编译C++代码
3.1 编译工程
g++ test.cpp -o test分解为以下4个steps
1、预处理-Pre-Processing (.i文件)
##- E 选项指示编译器仅对输入文件进行预处理g++ -E test.cpp -o test.i (.i 文件)
2、编译-Compiling (.s文件)
##-S 编译选项告诉 g++ 在为 c++ 代码产生了汇编语言文件后停止编译
##g++ 产生的汇编语言文件的缺省扩展名是 .sg++ -S test.i -o test.s (.s文件)
3、汇编-Assembling (.o文件)
##-c 选项告诉 g++ 仅把源代码编译为机器语言的目标代码
##缺省时 g++ 建立的目标代码文件有一个.o的扩展名g++ -S test.i -o test.s
4、链接-Linking (bin文件)
##-o 编译选项来为将产生的可执行文件用指定的文件名g++ test.o -o test
3.2 g++重要编译参数
1、-g 编译带调试信息的可执行文件
#-g 选项告诉GCC 产生能被GNU 调试器使用的调试信息,以调试程序
#产生带调试信息的可执行文件testg++ -g test.cpp
2、-O[n] 优化源代码
##所谓优化,例如省略掉代码中从未使用过的变量。直接将常量表达式用结果值代替等等,这些操作会缩减目标文件所包含的代码量,提高最终生成的可执行文件的运行效率
#-O 选项告诉 g++ 对源代码进行基本优化。这些优化在大多数情况下都会使程序执行的更快。-O2选项告诉g++ 产生尽可能小和尽可能快的代码。例如-O2、-O3、-On(n常为0~3)
#-O同时减小代码的长度和执行时间,其效果等价于-O1
#-O0表示不做优化,
#-O1为默认优化,
#-O2除了完成-O1的优化之外,还进行一些额外的调整工作,如指令日调整等。
#-O3则包括循环展开和其他一些与处理特性相关的优化工作
#选项使编译的速度比使用-O时慢,但通常产生的代码执行速度会更快#使用 -O2优化源代码,并输出可执行文件
g++ -O2 test.cpp
3、-l和-L指定库文件|指定库文件路径
##-l 参数(小写)就是用来指定程序要链接的库,-l 参数紧接着就是库名
##在/lib和/usr/lib和/usr/local/lib里的库直接用-l参数就能链接#链接glog库
g++ -lglog test.cpp
#如果库文件没放在上面三个目录里,需要使用-L参数(大写)指定库文件所在目录
#-L参数跟着的是库文件所在的目录名#链接 mytest库, libmytest.so在/home/bing/mytestlibfolder目录下
g++ -L/home/bing/mytestlibfolder(库路径) -lmytest(库名) test.cpp
4、**-i(大写)**指定头文件搜索目录
-i(大写)
##/usr/include目录一般是不用指定的,gcc知道去哪里找,但是如果头文件不在/usr/include里我们就要用-I 参数指定了,比如头文件放在/myinclude目录里,那编译命令行就要夹上-I/myinclude参数了,如果不加你就会得到一个“xxxx.h:No such file or directory”的错误。-I参数可以用相对路径,比如头文件在当前目录,就可以用-I .来指定。上面我们提到-cflags参数就是用来生成-I参数的。g++ -I/myinclude test.cpp
5、-Wall打印警告信息
打印出gcc提供的警告信息
g++ -Wall test.cpp
6、-w关闭警告信息
关闭所有警告信息
g++ -w test.cpp
很重要,如果编译标准不一样是不能编译成功的
7、-std=c++11设置编译标准
#使用 c++11 标准编译 test.cpp
g++ -std=c++11 test.cpp
8、-o 指定输出文件名
#指定即将产生的文件名
#指定输出可执行文件名为test
g++ test.cpp -o test (如果不指定的话会默认输出名为a.out)
9、- D定义宏
#在使用gcc/g++编译的时候定义宏
#常用场景:
-DDEBUG 定义DEBUG宏,可能文件中有DEBUG宏部分的相关信息,用个DDEBUG来选择开启或关闭DEBUG
示例代码:
// -Dname 定义宏name,默认定义内容为字符串“1”
#include <stdio.h>
int main()
{
#ifdef DEBUG
printf("DEBUG LOG\n");
#endif
printf("in\n");
}
// 1、在编译的时候,使用gcc -DDEBUG main.cpp
// 2、第七行代码可以被执行
注:使用man gcc命令可以查看gcc英文使用手册,如下图
3.1 【实战】g++命令行编译
案例:最初目录结构:2 directories,3files
** tree .**
3.3.1直接编译
最简单的编译,并运行
#将 main.cpp src/Swap.cpp编译为可执行文件
####若只输入g++ main.cpp src/Swap.cpp因为Swap.cpp没有和Swap.h在同一级,那么就会报错g++ main.cpp src/Swap.cpp -Iinclude
#运行a.out
./a.out
####增加参数编译,并运行
#将main./cpp src/Swap.cpp编译为可执行文件 附带一堆参数
g++ main.cpp src/Swap.cpp -Iinclude -std=c++11 -o2 -Wall -o b.out
#运行 b.out
./b.out
3.3.2 生成库文件并编译
链接 静态库 生成可执行文件1:
##进入src目录下
$cd src
#汇编,生成Swap.o文件
g++ Swap.cpp -C -I…/include
#生成静态库libSwap.a
ar rs libSwap.a Swap.o
##回到上级目录
$cd
##链接,生成可执行文件:staticmain
g++ main.cpp -Iinclude -Lsrc -lSwap -o staticmain
链接 动态库 生成可执行文件2:
##进入src目录下
$cd src
#生成动态库libSwap.so
g++ Swap.cpp -I…/include -fPIC -shared -o libSwap.so
##上面命令等价于一下两条命令
gcc Swap.cpp -I…/include -c -fPIC
gcc -shared -o libSwap.so Swap.o
##回到上级目录
$cd …
##链接,生成可执行文件:sharemain
g++ main.cpp -Iinclude -Lsrc -lSwap -o sharemain
-
-fPIC是地址无关码
-
-I (大写i):-I 后跟头文件目录,如-I ./lib/openssl/include。
-
-L(大写L):-L 后跟库文件目录,库文件为静态库.a或动态库.so文件,如 -L ./lib/sqlite。
-
-l (小写L):-I 后跟动态库名字,如果是静态库,要加上“-static”。
编译完成后的目录结构
最终目录结构:2 directories,8 files
3.3.3 运行可执行文件
运行可执行文件1
#运行可执行文件1
./staticmain
运行可执行文件2
#运行可执行文件2
LD_LIBRARY_PATH=src ./staticmain
因为动态库的可执行文件 dyna_main与依赖库 libswap,so不在一个目录中