本篇介绍什么是gcc,编译和链接都做了什么,以及gcc的参数-L 和 -l 的区别。
一、什么是gcc
gcc/g++是GNU的c/c++编译器,其对源程序.c执行编译和链接工作需要四步:
1.预处理,生成.i的文件[预处理器cpp]
2.将预处理后的文件不转换成汇编语言,生成文件.s[编译器egcs]
3.有汇编变为目标代码(机器代码)生成.o的文件[汇编器as]
4.连接目标代码,生成可执行程序[链接器ld]
二、编译和链接都干什么
gcc -c 源文件.c -------------------- 生成中间代码文件.o --------------------- 是进行“编译”工作。
gcc -o 目标文件 中间代码文件.o-------------------- 生成目标文件 ---------------------- 是进行“链接”工作。
编译时:编译器需要语法正确,函数与变量的声明正确。对于函数与变量,通常只需要告诉编译器头文件的所在位置(头文件中应该只是声明,而定义应该在c/c++文件中)。只要语法正确,编译器就可以编译出中间目标文件。
链接时:主要是链接函数和全局变量,所以我们使用这些中间目标文件(.o或.obj文件)来链接应用程序。链接器不管函数所在的源文件,只管函数的中间目标文件(一些静态库或系统库对应的中间目标文件是.so文件 .a文件),通过中间目标文件来找到函数的定义。所以如果有链接错误说:undefined symbol first referenced in file,表明gcc编译器没有找到其定义,需要把中间目标文件生成并链接过来。
三、gcc的两个参数-l 和 -L
在用到g_source_new等函数的时候,链接时出现如下错误:
gcc -o GSource GSource.o hbtime.o -I/usr/include/glib-2.0 -I/usr/lib/glib-2.0/include -L/usr/include/glib-2.0/glib -I/usr/lib/glib-2.0/glib/include -lpthreadUndefined first referenced
symbol in file
main /usr/lib/crt1.o
g_source_set_can_recurse GSource.o
g_source_set_priority GSource.o
g_main_context_find_source_by_id GSource.o
g_source_set_callback GSource.o
g_source_remove GSource.o
g_source_unref GSource.o
g_source_new GSource.o
g_source_attach GSource.o
ld: fatal: symbol referencing errors. No output written to GSource
collect2: ld returned 1 exit status
*** Error code 1
make: Fatal error: Command failed for target `GSource' 其中,左面有关g_source_*的函数声明均在usr/include/glib-2.0/glib路径下的gmain.h中,定义未显示找到,但估计是在/usr/lib/libglib-2.0.so里。于是我通过-L参数将其路径包含进来,make了一下,还是出现上述错误。后经调研测试发现,没有执行真正的链接操作(-lglib-2.0),我只是把中间目标文件so的位置找到了,告诉了gcc编译器. 哎,教训啊! 两个参数的含义为: -llibrary
制定编译的时候使用的库
例子用法
gcc -lpthread hello.c
使用pthread库编译程序
-Ldir
制定编译的时候,搜索库的路径。比如你自己的库,可以用它制定目录,不然
编译器将只在标准库的目录找。这个dir就是目录的名称。 也就是说,-L是找位置,-l是根据找到的位置和库进行具体的链接操作。