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

               

作者:朱金灿

来源:http://blog.csdn.net/clever101

 

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

  undefined reference to `inflateInit_'|

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

           

                      

     

        修改为下图:

                     

       

         即把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.cint main(){        extern int f1( int , int );        extern int f2( int , int );        return f1( 1 , 2 ) + f2( 3, 4 );}---- f1.cint f1( int a , int b ) { return a + b; }---- f2.cextern int f3( int , int );int f2( int a , int b ) { return a * b - f3( a , b ); }---- f3.cint 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大侠!





 


           

再分享一下我老师大神的人工智能教程吧。零基础!通俗易懂!风趣幽默!还带黄段子!希望你也加入到我们人工智能的队伍中来!https://blog.csdn.net/jiangjunshow

猜你喜欢

转载自blog.csdn.net/gdruhv/article/details/86522375