一、头文件与库文件的区别
- 头文件一般而言,是申明和定义。
- 库文件是已经编译好的二进制代码。这个二进制代码可以是动态的,如 .so ;也可以是静态的,如 .a 。如果是动态的,则最后生成的程序文件在运行时,需要这个动态库的支持;如果是静态的,则最后生成的可执行程序文件运行时可以脱离这个库文件而独立运行。
头文件以.h结尾,可以用文本编辑器查看内容。是ASCII的。
而库文件以.a(静态库)或.so(动态库)结尾,是二进制的。
二、从几个常遇到的问题谈起
问题一:undefined reference to 'xxx'.
问题二:/usr/bin/ld:cannot find -lxxx.
问题三:xxx.h:No such file or directory.
首先,这几个问题都不是编译错误,是链接错误,也就是如果出现的是这几个错误,说明你的源程序本身没有问题,是你的编译选项用的不对或者缺少相关的库文件或者头文件。前两个问题是找不到库文件的问题,后一个问题是找不到头文件的问题。下面详细说一下头文件和库文件相关的问题。
三、库文件相关(-l选项和-L选项(大小写L))
编译完成之后就进入链接阶段,这里就涉及到函数库,比如通常的用的 printf 函数,我们仅仅在程序开始包含进了“stdio.h”,这个里面也只有该函数的声明,而没有定义函数的实现,那么,printf函数的实现在哪里呢?答案是在函数库中,链接时,gcc会链接到具体的函数库中,在那里可以找到printf函数的实现。
-l 参数就是用来指定程序要链接的库,-l 参数紧接着就是库名,那么库名跟真正的库文件名有什么关系呢?就拿数学库来说,他的库名是 m ,他的库文件名是 libm.so,很容易看出,把库文件名的头 lib 和尾 .so 去掉就是库名了。
好了现在我们知道怎么得到库名了,比如我们自已要用到一个第三方提供的库名字叫libtest.so,那么我们只要把libtest.so拷贝到/usr/lib里,编译时加上-ltest参数,我们就能用上libtest.so库了(当然要用libtest.so库里的函数,我们还需要与libtest.so配套的头文件)。
放在 /lib 和 /usr/lib 和 /usr/local/lib 里的库可直接用 -l 参数就能链接了。但如果库文件没放在这三个目录里,而是放在其他目录里,这时我们只用 -l 参数的话,链接还是会出错,出错信息大概是:“/usr/bin/ld:cannot find -lxxx”,也就是链接程序 ld 在那3个目录里找不到 libxxx.so,这时另外一个参数-L就派上用场了,比如常用的X11的库,它放在/usr/X11R6/lib目录下,我们编译时就要用-L/usr/X11R6/lib -lX11参数,-L参数跟着的是库文件所在的目录名。再比如我们把libtest.so放在/aaa/bbb/ccc目录下,那链接参数就是-L/aaa/bbb/ccc -ltest。
四、头文件相关(-I选项(大写i))
-I 参数是用来指定头文件目录,/usr/include 目录一般是不用指定的,gcc知道去那里找,但是如果头文件不在 /usr/include 里我们就要用 -I 参数指定了,比如头文件放在 /myinclude 目录里,那编译命令行就要加上-I/myinclude参数了,如果不加你会得到一个"xxxx.h: No such file or directory"的错误。-I 参数可以用相对路径,比如头文件在当前目录,可以用-I.来指定。