动态链接的无提示重复符号错误

时间:2012-10-25 22:39:54

标签: linker shared-libraries

我有以下C文件:

Base.h:

void g();
void h();

Base.c:

#include <stdio.h>
#include <Base.h>

void g() {
  printf("This is lib g\n");
  h();
}

void h() {
  printf("This is lib h\n");
}

A.C:

#include <stdio.h>
#include <Base.h>

void h() {
  printf("This is A h\n");
}

void main() {
  g();
  h();
}

我按如下方式编译和链接:

$ gcc -c -fPIC -o Base.o -I. Base.c
$ gcc -shared -o libBase.so Base.o
$ gcc -c -o A.o A.c
$ gcc -o A A.o -lBase -L.

现在我运行程序

$ LD_LIBRARY_PATH=. ./A

并获得:

This is lib g
This is A h
This is A h

这意味着,libBase中对h的调用由A.o解决。这不是我的预期。我要么期望动态链接器在libBase中使用h解析对lib的调用,或者在第四次gcc调用中解析错误消息。

如果我将A.c中的h重命名为h1

#include <stdio.h>
#include <Base.h>

void h1() {
  printf("This is A h1\n");
}

void main() {
  g();
  h1();
}

我获得了

This is lib g
This is lib h
This is A h1

所以,在这种情况下,h按照我的预期得到解决。

我需要做些什么才能获得错误消息或将g中的调用解析为libBase中的h?

1 个答案:

答案 0 :(得分:2)

  

这不是我的预期。

你的期望是错误的。这是大多数UNIX系统上共享库的工作方式:加载器只是在加载库列表中,并尝试查找给定的符号。第一个定义符号“wins”的库。

这种行为有很多优点。例如:您可以LD_PRELOAD libtcmalloc.so,突然所有mallocfree来电都会解析为tcmalloc

在ELF系统上,您可以使用-Bsymbolic链接器标志修改此行为(在通过GCC时使用-Wl,-Bsymbolic)。请注意:-Bsymbolic“违反系统”,因此您可能会遇到许多意外和不良副作用。

另见this答案。