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

War*_*ren 19 c gcc shared-libraries ld

为什么我使用gcc得到"未定义的引用"错误?

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

文件:external.c

int external() {
    return 5;
}
Run Code Online (Sandbox Code Playgroud)

文件:program.c

int external();
int main(char** argv, int* argc) {
    return external();
}
Run Code Online (Sandbox Code Playgroud)

命令:

$ 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
Run Code Online (Sandbox Code Playgroud)

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

命令:

$ nm libexternal.so | grep external
0000040c T external
Run Code Online (Sandbox Code Playgroud)

我在这里错过了什么?

Fle*_*exo 37

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

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

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

  • 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,如果一个或两个program.c/libexternal.so导致需要,它将始终链接到外部,但仅链接到mightneed .

  • 感谢您添加"--as-needed/ - no-as-needed"解释.这正是我所寻找的. (2认同)