为什么我得到一个gcc“未定义的引用”错误试图创build共享对象?

为什么使用gcc得到“未定义参考”错误?

我正在尝试创build一个导出一个函数“external()”的共享对象(.so)。 然后我尝试链接到.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正在定义“外部”:

命令:

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

我在这里错过了什么?

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

这意味着,如果在C文件之前写入-lexternal ,库将自动被排除(在testing是否需要这样的事情时,顺序很重要)

您可以通过以下两种方法解决此问题

  • 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 gcc -L. -Wl,--no-as-needed -lexternal -Wl,--as-needed -o program program.c -lmightneed将始终与外部链接,但只有在链接的时候才会受到威胁, 如果其中一个或两个都是program.c / libexternal。所以造成这是需要的。