参考博客:
http://www.cnblogs.com/qytan36/archive/2010/05/25/1743955.html
http://m.blog.csdn.net/article/details?id=24421919
一、GCC编译C源码有四个步骤:预处理-----> 编译 ----> 汇编 ----> 链接
四阶段作用:(对应文件后缀依次是:.i>.s>.o>.exe)
》预处理 将引进的头文件的代码加入到当前文件中。选项"-E"预处理。 用法:gcc -E hello.c –o hello.i
》编译 Gcc首先要检查代码的规范性、是否有语法错误等,在检查无误后,Gcc把代码翻译成汇编语言。选项"-S"预处理。 用法:gcc -S hello.i –o hello.s.(该选项只进行编译而不进行汇编,生成汇编代码。)
》汇编 汇编阶段是把编译阶段生成的”.s”文件转成二进制目标代码。选项"-c"预处理。 用法:gcc -c hello.s –o hello.o。
》链接 在成功编译之后,就进入了链接阶段。无选项链接 用法:gcc hello.o –o hello.exe。
注释:
在链接阶段,编译器其实是将系统标准头文件对应的实现体编译所称的对应的链接库,如stdio.h对应的名为libc.so.6的库文件,链接进入自己汇编后生成的.o文件中。最终生成可执行程序。
可以用ldd命令查看动态库加载情况。
函数库一般分为静态库和动态库两种。静态库是指编译链接时,把库文件的代码全部加入到可执行文件中,因此生成的文件比较大,但在运行时也就不再需要库文件了。其后缀名一般为”.a”。动态库与之相反,在编译链接时并没有把库文件的代码加入到可执行文件中,而是在程序执行时由运行时链接文件加载库,这样可以节省系统的开销。动态库一般后缀名为”.so”,如前面所述的libc.so.6就是动态库。gcc在编译时默认使用动态库。
二、关于GCC/G++ 重要参数,及与动态库和静态库。关于两者的区别请参考我之前的文章:关于动态链接库与静态链接库
1. -I 编译阶段指定依赖库的头文件路径,便于找到其头文件,此阶段只编译,不链接。
2. -L libpath -llibname 连接阶段指定依赖的库(静态库/动态库)的名字、存放路径。
3. 编译静态库时,只编译不链接,所以如果该库依赖其他库时,最后生成执行程序时需要链接就会报错
4. 编译动态库时,即编译,又链接,运行时加载动态库
5. 对于大的项目,系统自己的库文件、第三方库文件比较多
生成执行程序时,链接库时,是按照从右向左编译-l指定的库的方式编译的,如果这些参数顺序与实际的依赖关系不符,则链接出现"undefined reference to ×××"这样的错误。
建议梳理清楚库之间的依赖关系,被依赖的库放在右边,这样就可以避免此等问题的出现。
当然也可以让编译器根据程序的依赖关系自动编译 -Xlinker "-(" -lmcsdbacc -lmcslibs -lmcsbase -lmsgptc -Xlinker "-)" 这样就避免依赖关系与编译参数关系不符导致链接错误,但是同时会增加编译时间;