gcc链接静态库时对待.a文件和.o文件的不同

时间:2021-10-30 01:58:38

  很多人都知道,gcc在链接静态库时是从前往后找符号。因此如果一份文件foo引用了静态库bar.a,那么在链接命令中,bar.a必须放在foo的后面,也就是像gcc ... foo ... bar.a这样;否则链接时会报找不到定义的错误(即undefined reference to ...)。

 

  .a文件其实没什么特别的地方,它不过是将多个.o文件打包成一份文件。如果我们在链接命令中,直接用.o文件替换.a文件,那也需要遵循gcc的这种链接顺序吗?可以用gcc .. bar.o ... foo这样的链接命令吗?可以做个简单的试验。

 

  假设我们有两份源代码文件。一份是foo.c,定义了一个foo函数。

void foo()
{
}

  另一份是main.c,里面引用了foo函数:

void foo();

int main()
{
foo();
}

  我们将foo.c编译成foo.o,然后打包成foo.a:

$ gcc -c foo.c

$ ar rcs foo.a foo.o

  如果我们让main.c链接foo.a,并且将foo.a放在main.c的前面,那就会如期望的那样出现链接错误:

$ gcc -o test foo.a main.c
C:\Users\ADMINI~1\AppData\Local\Temp\ccan3qBI.o:main.c:(.text+0xc): undefined reference to `foo'
collect2.exe: error: ld returned 1 exit status

  但如果将foo.a换成foo.o,链接就不会报错了:

$ gcc -o test foo.o main.c

  所以结论是:.o文件在链接中并不遵循.a文件那样的链接顺序,看来链接器找符号时,一定会搜索命令中的.o文件。

 

  *里的这篇帖子也提到了类似的事:http://*.com/questions/11231101/gcc-linking-object-file-and-library-what-is-the-difference

 

  试验用的gcc版本:

$ gcc -v
Using built-in specs.
COLLECT_GCC=e:\Zhixiang\MinGW\bin\gcc.exe
COLLECT_LTO_WRAPPER=e:/zhixiang/mingw/bin/../libexec/gcc/mingw32/4.8.1/lto-wrapper.exe
Target: mingw32
Configured with: ../gcc-4.8.1/configure --prefix=/mingw --host=mingw32 --build=mingw32 --without-pic --enable-shared --enable-static --with-gnu-ld --enable-lto --enable-libssp --disable-multilib --enable-languages=c,c++,fortran,objc,obj-c++,ada --disable-sjlj-exceptions --with-dwarf2 --disable-win32-registry --enable-libstdcxx-debug --enable-version-specific-runtime-libs --with-gmp=/usr/src/pkg/gmp-5.1.2-1-mingw32-src/bld --with-mpc=/usr/src/pkg/mpc-1.0.1-1-mingw32-src/bld --with-mpfr= --with-system-zlib --with-gnu-as --enable-decimal-float=yes --enable-libgomp --enable-threads --with-libiconv-prefix=/mingw32 --with-libintl-prefix=/mingw --disable-bootstrap LDFLAGS=-s CFLAGS=-D_USE_32BIT_TIME_T
Thread model: win32
gcc version 4.8.1 (GCC)