一、GCC简介
-
预处理(Pre-Processing):生成 .i 的文件[预处理器cpp]
-
编译(Compiling): 将预处理后的文件转换成汇编语言, 生成文件 .s [编译器egcs]
-
汇编(Assembling): 由汇编变为目标代码(机器代码)生成 .o 的文件[汇编器as]
-
链接生成可执行文件(Linking) [链接器ld]
文章来源:https://www.toymoban.com/news/detail-464155.html
# 安装c编译器(linux需安装python-devel, gcc)
centos: yum install python-devel gcc
ubuntu: apt-get install build-essential
二、GCC使用
nist@zq-node2:~/test$ gcc --help
Usage: gcc [options] file...
Options:
-pass-exit-codes 从一个阶段以最高错误代码退出.
--help Display this information.
--target-help 显示特定于目标的命令行选项.
--help={common|optimizers|params|target|warnings|[^]{joined|separate|undocumented}}[,...].
显示特定类型的命令行选项.
(使用 '-v --help' 显示子进程的命令行选项).
--version 显示编译器版本信息.
-dumpspecs 显示所有内置规范字符串.
-dumpversion 显示编译器的版本.
-dumpmachine 显示编译器的目标处理器.
-print-search-dirs 显示编译器搜索路径中的目录.
-print-libgcc-file-name 显示编译器配套库的名称.
-print-file-name=<lib> 显示库 <lib> 的完整路径.
-print-prog-name=<prog> 显示编译器组件 <prog> 的完整路径.
-print-multiarch 显示目标的规范化 GNU 三元组,用作库路径中的一个组件.
-print-multi-directory 显示 libgcc 版本的根目录.
-print-multi-lib 显示命令行选项和多个库搜索目录之间的映射.
-print-multi-os-directory 显示操作系统库的相对路径.
-print-sysroot 显示目标库目录.
-print-sysroot-headers-suffix 显示用于查找headers的sysroot后缀.
-Wa,<options> 将逗号分隔的 <options> 传递给汇编器(assembler).
-Wp,<options> 将逗号分隔的 <options> 传递给预处理器(preprocessor)
-Wl,<options> 将逗号分隔的 <options> 传递给链接器(linker)
-Xassembler <arg> 将 <arg> 传递给汇编器(assembler).
-Xpreprocessor <arg> 将 <arg> 传递给预处理器(preprocessor).
-Xlinker <arg> 将 <arg> 传递给链接器(linker).
-save-temps 不用删除中间文件.
-save-temps=<arg> 不用删除指定的中间文件.
-no-canonical-prefixes 在构建其他 gcc 组件的相对前缀时,不要规范化路径.
-pipe 使用管道而不是中间文件
-time 为每个子流程的执行计时.
-specs=<file> 使用 <file> 的内容覆盖内置规范.
-std=<standard> 假设输入源用于<standard>。
--sysroot=<directory> 使用<standard>作为headers和libraries的根目录.
-B <directory> 将 <directory> 添加到编译器的搜索路径.
-v 显示编译器调用的程序.
-### 与 -v 类似,但引用的选项和命令不执行.
-E 仅执行预处理(不要编译、汇编或链接).
-S 只编译(不汇编或链接).
-c 编译和汇编,但不链接.
-o <file> 指定输出文件. gcc 编译出来的文件默认是 a.out
-pie 创建一个动态链接、位置无关的可执行文件
-shared 创建共享库/动态库.
-g: 生成调试信息
-w: 不生成任何警告
-Wall: 编译时 显示Warning警告,但只会显示编译器认为会出现错误的警告
-x <language> 指定以下输入文件的语言。允许的语言包括:c c++汇编程序none“none”表示恢复到的默认行为根据文件的扩展名猜测语言。
Options starting with -g, -f, -m, -O, -W, or --param are automatically
passed on to the various sub-processes invoked by gcc. In order to pass
other options on to these processes the -W<letter> options must be used.
For bug reporting instructions, please see:
<file:///usr/share/doc/gcc-9/README.Bugs>.
-shared:
编译动态库时要用到
-pthread:
在Linux中要用到多线程时,需要链接pthread库
-fPIC:
作用于编译阶段,告诉编译器产生与位置无关代码(Position-Independent Code),
则产生的代码中,没有绝对地址,全部使用相对地址,故而代码可以被加载器加载到内存的任意
位置,都可以正确的执行。这正是共享库所要求的,共享库被加载时,在内存的位置不是固定的。
-fwrapv:
它定义了溢出时候编译器的行为——采用二补码的方式进行操作
-O参数
这是一个程序优化参数,一般用-O2就是,用来优化程序用的
-O2:
会尝试更多的寄存器级的优化以及指令级的优化,它会在编译期间占用更多的内存和编译时间。
-O3: 在O2的基础上进行更多的优化
-Wall:
编译时 显示Warning警告,但只会显示编译器认为会出现错误的警告
-fno-strict-aliasing:
“-fstrict-aliasing”表示启用严格别名规则,“-fno-strict-aliasing”表示禁用严格别名规则,当gcc的编译优化参数为“-O2”、“-O3”和“-Os”时,默认会打开“-fstrict-aliasing”。
-I (大写的i):
是用来指定头文件目录
-I /home/hello/include表示将/home/hello/include目录作为第一个寻找头文件的目录,寻找的顺序是:/home/hello/include-->/usr/include-->/usr/local/include
-l:
-l(小写的 L)参数就是用来指定程序要链接的库,-l参数紧接着就是库名,把库文件名的头lib和尾.so去掉就是库名了,例如我们要用libtest.so库库,编译时加上-ltest参数就能用上了
2.1、gcc编译的四个步骤
预处理: gcc -E Test.c -o Test.i # -E选项,可以让编译器在预处理后停止,并输出预处理结果
编译: gcc -S Test.i -o Test.s # -S选项,表示在程序编译期间,将我们的代码编译成汇编语言。
汇编: gcc -c Test.s -o Test.o # -c选项,表示由汇编器负责将刚才的.s文件编译为目标文件,即计算机所能识别的序列。
链接生成可执行文件: gcc Test.o -o Test # 将刚才的Test.o文件与C标准输入输出库进行连接,最终生成程序Test可执行文件
2.2、多源文件的编译方法
gcc testfun.c test.c -o test
作用:将testfun.c和test.c分别编译后链接成test可执行文件。文章来源地址https://www.toymoban.com/news/detail-464155.html
gcc -c testfun.c -o testfun.o # 将testfun.c编译成testfun.o
gcc -c test.c -o test.o # 将test.c编译成test.o
gcc -o testfun.o test.o -o test # 将testfun.o和test.o链接成test
2.3、库文件连接
-
ld会去找GCC命令中的参数-L
-
再找gcc的环境变量LIBRARY_PATH
-
再找内定目录 /lib、 /usr/lib、 /usr/local/lib 这是当初compile gcc时写在程序内的
-
编译目标代码时指定的动态库搜索路径
-
环境变量LD_LIBRARY_PATH指定的动态库搜索路径
-
配置文件/etc/ld.so.conf中指定的动态库搜索路径
-
默认的动态库搜索路径/lib
-
默认的动态库搜索路径/usr/lib
-
LIBRARY_PATH环境变量:指定程序静态链接库文件搜索路径
-
LD_LIBRARY_PATH环境变量:指定程序动态链接库文件搜索路径
三、示例
3.1、阶段编译
#include <stdio.h>
int main(void)
{
printf("Hello, GetIoT\n");
return 0;
}
gcc hello.c
gcc hello.c -o hello
gcc -E hello.c -o hello.i
gcc -S hello.c -o hello.s
gcc -S hello.i -o hello.s
gcc -c hello.c -o hello.o
gcc -c hello.i -o hello.o
gcc -c hello.s -o hello.o
gcc hello.o -o hello
3.2、使用静态库
#include <stdio.h>
void foo(void)
{
printf("Here is a static library\n");
}
gcc -c foo.c # 生成 foo.o 目标文件
ar rcs libfoo.a foo.o # 生成 libfoo.a 静态库
#include <stdio.h>
void foo(void);
int main(void)
{
printf("Hello, GetIoT\n");
foo();
return 0;
}
gcc hello.c -static libfoo.a -o hello
gcc hello.c -static -L. -lfoo -o hello
3.3、使用共享库
#include <stdio.h>
void foo(void)
{
printf("Here is a shared library\n");
}
gcc foo.c -shared -fPIC -o libfoo.so
#include <stdio.h>
void foo(void);
int main(void)
{
printf("Hello, GetIoT\n");
foo();
return 0;
}
gcc hello.c libfoo.so -o hello
gcc hello.c -L. -lfoo -o hello
$ ./hello./hello: error while loading shared libraries: libfoo.so: cannot open shared object file: No such file or directory
$ ldd hello
linux-vdso.so.1 (0x00007fff5276d000)
libfoo.so => not found
libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007fcc90fa7000)
/lib64/ld-linux-x86-64.so.2 (0x00007fcc911bd000)
export LD_LIBRARY_PATH=$(pwd)
gcc hello.c -L. -lfoo -Wl,-rpath=`pwd` -o hello
sudo cp libfoo.so /usr/lib/
到了这里,关于gcc编译的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!