作者:朱金灿

 

          使用CodeBlocks10.05编一个小程序用到了png库和zlib库。我发现编译png静态库时,只需要指定zlib库的头文件路径,但并不需要链接到zlib库(这个我感觉有点奇怪)。 然后编译自己写的程序时出现一个错误:


  undefined reference to `inflateInit_'|


        inflateInit是zlib库的一个函数,这里编译错误显示为inflateInit_(我估计是gcc给函数添加了修饰符的缘故)。我检查了我的包含头文件路径和lib文件,都设置好了,为何会出现这个错误呢?后来很偶然的想到是不是链接库的顺序问题,就改了下顺序,将下图的:

从一个链接错误探究GCC的链接库顺序_链接库

        修改为下图:

                     

从一个链接错误探究GCC的链接库顺序_file_02         即把png库提到zlib库的前面然后重新编译这个编译错误就消失了。这是为什么呢?


       上论坛求教,mLee79大侠告诉我:gcc 从前到后在各个符号,找不到就报错, 又不会往前面去找。如果不想安排链接顺序,就在编译选项添加 -Xlinker "-("$(LIBS) -Xlinker "-)"。一个简单的例子是:


$ for file in app.c f1.c f2.c f3.c ; do echo ---- $file ; cat $file ; done ;  gcc -c f1.c f2.c f3.c ; \
ar crf lib1.a f1.o f3.o ; ar crf lib2.a f2.o ; gcc app.c lib1.a lib2.a ; \
gcc app.c -Xlinker "-(" lib1.a lib2.a -Xlinker "-)"
---- app.c
int main()
{
extern int f1( int , int );
extern int f2( int , int );
return f1( 1 , 2 ) + f2( 3, 4 );
}
---- f1.c
int f1( int a , int b ) { return a + b; }
---- f2.c
extern int f3( int , int );
int f2( int a , int b ) { return a * b - f3( a , b ); }
---- f3.c
int f3( int a , int b ) { return a - b; }

lib2.a(f2.o):f2.c:(.text+0x1e): undefined reference to `_f3'
collect2: ld returned 1 exit status



        这里简单说明下这个例子:程序首先链接的lib1(f3符号在lib1),当程序链接到lib2时,很自然就找不到f3(因为lib1在lib2的前面),


        不过可以想象加 -Xlinker肯定很慢,因为每个符号都要查找所有的库。因此要解决此类链接错误还是得安排链接库的顺序,遵循的原则是上层调用库放在底层被调用库的前面。在此非常感谢mLee79大侠!