linux oracle 网络安全 编程
分类: C/C++
2014-01-14 15:39:46
头文件以.h结尾,可以用文本编辑器查看内容。是ASCII的。 而库文件以.a(静态库)或.so(动态库)结尾,是二进制的。 |
问题一:undefined reference to 'xxx'.
问题二:/usr/bin/ld:cannot find -lxxx.
问题三:xxx.h:No such file or directory.
首先,这几个问题都不是编译错误,是链接错误,也就是如果出现的是这几个错误,说明你的源程序本身没有问题,是你的编译选项用的不对或者缺少相关的库文件或者头文件。前两个问题是找不到库文件的问题,后一个问题是找不到头文件的问题。下面详细说一下头文件和库文件相关的问题。
编译完成之后就进入链接阶段,这里就涉及到函数库,比如通常的用的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参数是用来指定头文件目录,/usr/include目录一般是不用指定的,gcc知道去那里找,但是如果头文件不在/usr/include里我们就要用-I参数指定了,比如头文件放在/myinclude目录里,那编译命令行就要加上-I/myinclude参数了,如果不加你会得到一个"xxxx.h: No such file or directory"的错误。-I参数可以用相对路径,比如头文件在当前目录,可以用-I.来指定。
手动来写这些编译选项参数一般比较麻烦,而且容易出错,比如,我程序中用到了glib库中的函数,由于glib库一般不是系统自带的库,我是手动编译源码安装的,默认安装路径是/usr/local/lib,头文件在/usr/local/include/glib-2.0下面。我编译我自己的代码时,如果直接编译,不加任何编译选项,会提示"glib.h: No such file or directory"错误,这里有两个解决方法。
方法一:第一安装的时候手动指定安装路径,安装到/usr下面,默认安装是直接”./configure”, “make”和”make install”三条命令,如果要手动指定安装路径就用”./configure --prefix=/usr”,”make”和”make install”.
方法二:编译的时候添加编译选项。我这里没有直接用-l和-I选项,我利用了一个工具pkg-config。具体的编译命令是:“gcc`pkg-config --cflags --libs glib-2.0 ` test.c”。
pkg-config会自动的生成指定库的库文件和头文件路径,pkg-config的用法就是“ pkg-config --cflags --libs pkgName”, 其中pagName是包名,如果不知道具体的包名是什么,可以通过命令“pkg-config --list-all”查询,该命令会列出系统支持的所有开发包。