gcc 6个月前

编程语言
887
gcc

在文章开篇前言有声名本Linux系列主要偏向C/C++进行教学

一、安装gcc

#Ubuntu
sudo apt update   #更新本地的软件下载列表, 得到最新的下载地址
sudo apt install gcc g++

#centos
sudo yum update 
sudo yum install gcc g++

之所以更新下载列表,是因为这样可以下载最新的gcc g++ 以支持c++11

#两种方法查看gcc g++版本
gcc -v
gcc --version

g++ -v
g++ --version

最低4.8.5的版本才支持C++11


二、gcc工作流程


gcc工作分为四步 1.预处理: 在这个阶段主要做了三件事: 展开头文件 、宏替换 、去掉注释行 这个阶段需要GCC调用预处理器来完成, 最终得到的还是源文件, 文本格式 2.编译: 这个阶段需要GCC调用编译器对文件进行编译, 最终得到一个汇编文件 3.汇编: 这个阶段需要GCC调用汇编器对文件进行汇编, 最终得到一个二进制文件 4.链接: 这个阶段需要GCC调用链接器对程序需要调用的库进行链接, 最终得到一个可执行的二进制文件

文件名后缀 说明 gcc参数
.c 源文件
.i 预处理后的 C 文件 -E
.s 编译之后得到的汇编语言的源文件 -S
.o 汇编后得到的二进制文件 -c

如果要编译单个文件非常简单,只需要 gcc c源文件 会自动生成文件名为a.out的可执行文件(若已经存在会覆盖) 也可以通过 -o 参数指定生成的文件名

以下是4个步骤单独执行的方式

#对源文件进行预处理,gcc使用-E参数
#1. 预处理,-o指定生成的文件名
gcc -E test.c -o test.i

#2. 编译,得到汇编文件
gcc -S test.i -o test.s

#3.汇编
gcc -c test.s -o test.o

#4.链接
gcc test.o -o test

如果直接执行汇编或者链接,前面几步会自动执行

# 参数 -c 是进行文件的汇编, 汇编之前的两步会自动执行
gcc test.c -c -o app.o

# 该命令是直接进行链接生成可执行程序, 链接之前的三步会自动执行
gcc test.c -o app

三、gcc常用参数


参数在gcc命令中并没有位置要求,指定即可 | gcc编译选项 | 选项的意义 | |----------------|--------------------------------------------| | -E | 预处理指定的源文件,不进行编译 | | -S | 编译指定的源文件,但是不进行汇编 | | -c | 编译、汇编指定的源文件,但是不进行链接 | | -o [file1] [file2] / [file2] -o [file1] | 将文件 file2 编译成文件 file1 | | -I directory | 指定 include 包含文件的搜索目录 | | -g | 在编译的时候,生成调试信息,该程序可以被调试器调试 | | -D | 在程序编译的时候,指定一个宏 | | -w | 不生成任何警告信息, 不建议使用, 有些时候警告就是错误 | | -Wall | 生成所有警告信息 | | -On | n的取值范围:0~3。编译器的优化选项的4个级别 | | -l | 在程序编译的时候,指定使用的库 | | -L | 指定编译的时候,搜索的库的路径。 | | -fPIC/fpic | 生成与位置无关的代码 | | -shared | 生成共享目标文件。通常用在建立共享库时 | | -std | 指定C方言,如:-std=c99,gcc默认的方言是GNU C |


1.指定生成的文件名(-o)

# 参数 -o的用法 , 原材料 test.c 最终生成的文件名为 app
# test.c 写在 -o 之前
$ gcc test.c -o app

# test.c 写在 -o 之后
gcc -o app test.c

2.搜索头文件(-I)

如果在程序中包含了一些头文件, 但是包含的一些头文件在程序预处理的时候因为找不到无法被展开,导致程序编译失败,这时候我们可以在gcc命令中添加 -I参数重新指定要引用的头文件路径, 保证编译顺利完成。

gcc *.c -o calc -I ./include #编译当前目录内所有的c程序,命名为calc 
#头文件在include文件中,指明头文件所在地
#如果头文件就在当前目录中不加-I也可以执行

3.指定一个宏(-D)

如果不想在程序中定义某个宏, 但是又想让它存在,通过gcc的参数 -D就可以实现,编译器会认为参数后边指定的宏在程序中是存在的。

gcc test.c -o app -D DEBUG #编译中定义DEBUG这个宏

-D 参数的应用场景: 在发布程序的时候, 一般都会要求将程序中所有的log(日志)输出去掉, 如果不去掉会影响程序的执行效率,删除这些打印log的源代码是很麻烦的 解决方案是这样的: 将所有的打印log的代码都写到一个宏判定中, 可以模仿上边的例子 在编译程序的时候指定 -D 就会有log输出 在编译程序的时候不指定 -D, log就不会输出

四、多文件编译

gcc是可以多文件一起编译的

#生成可执行程序test
gcc -o test string.c main.c

#执行test
./test
#先将源文件编成目标文件,然后进行链接得到可执行程序
# 汇编生成二进制目标文件, 指定了 -c 参数之后, 源文件会自动生成 string.o 和 main.o
$ gcc –c string.c main.c

# 链接目标文件, 生成可执行程序 test
$ gcc –o test string.o main.o

# 运行可执行程序
$ ./test

五、gcc与g++


以下是gcc和g++的区别

  1. 在代码编译阶段(第二个阶段): 后缀为 .c 的,gcc 把它当作是C程序,而 g++ 当作是 C++ 程序 后缀为.cpp的,两者都会认为是 C++ 程序,C++ 的语法规则更加严谨一些 g++会调用gcc,对于C++代码,两者是等价的, 也就是说 gcc 和 g++ 都可以编译 C/C++代码
  2. 在链接阶段(最后一个阶段): gcc 和 g++ 都可以自动链接到标准C库 g++ 可以自动链接到标准C++库, gcc如果要链接到标准C++库需要加参数 -lstdc++
  3. 关于 __cplusplus宏的定义 g++ 会自动定义__cplusplus宏,但是这个不影响它去编译C程序 gcc 需要根据文件后缀判断是否需要定义 __cplusplus 宏 (规则参考第一条)

综上所述: 不管是 gcc 还是 g++ 都可以编译 C 程序,编译程序的规则和参数都相同 g++可以直接编译C++程序, gcc 编译 C++程序需要添加额外参数 -lstdc++ 不管是 gcc 还是 g++ 都可以定义 __cplusplus宏

#编译 c 程序
gcc test.c -o test    # 使用gcc
g++ test.c -o test    # 使用g++

#编译 c++ 程序
gcc test.cpp -lstdc++ -o test     # 使用gcc
g++ test.cpp -o test              # 使用g++
image
EchoEcho官方
无论前方如何,请不要后悔与我相遇。
1377
发布数
439
关注者
2243416
累计阅读

热门教程文档

Dart
35小节
Gin
17小节
Docker
62小节
CSS
33小节
C++
73小节