为什么我在尝试创建共享对象时遇到gcc“未定义引用”错误?

时间:2011-11-15 17:26:20

标签: c gcc ld shared-libraries

为什么我使用gcc收到“未定义的引用”错误?

我正在尝试创建一个导出一个函数的共享对象(.so),“external()”。然后我尝试链接.so但得到“未定义的引用'外部'”。我在这里做错了什么?

文件:external.c

int external() {
    return 5;
}

文件:program.c

int external();
int main(char** argv, int* argc) {
    return external();
}

命令:

$ gcc -fPIC -c external.c
$ gcc -shared -o libexternal.so external.o
$ gcc -L. -lexternal -o program program.c
/tmp/cc3MmhAE.o: In function `main':
program.c:(.text+0x7): undefined reference to `external'
collect2: ld returned 1 exit status

我甚至可以运行nm并看到.so正在定义'external':

命令:

$ nm libexternal.so | grep external
0000040c T external

我在这里缺少什么?

1 个答案:

答案 0 :(得分:36)

gcc / ld的最新版本默认链接--as-needed

这意味着如果您在C文件之前写-lexternal,库将自动被排除(测试时,如果“需要”这样的话,这个顺序很重要)

您可以使用以下任意一种解决此问题:

  • gcc -L. -o program program.c -lexternal
  • gcc -L. -Wl,--no-as-needed -lexternal -o program program.c

后者将--no-as-needed传递给链接器,这会导致库仍然被链接,即使您没有从中调用external()

注意:-Wl,--no-as-needed未全局应用于链接的所有内容,它仅适用于命令行顺序中的所有内容。所以-lexternal -Wl,--no-as-needed也行不通。这确实意味着您可以混合和匹配行为,例如,gcc -L. -Wl,--no-as-needed -lexternal -Wl,--as-needed -o program program.c -lmightneed将始终链接到外部,但仅链接到mightneed 如果 program.c / libexternal.so中的一个或两个它需要。