首页 > 解决方案 > ld:未定义的引用,但它应该让它们未解决

问题描述

我无法生成与另一个库 (SDL) 一起使用的库。我正在使用 MinGW 进行制作,并使用 ld 进行链接。我很困惑,因为a)它不应该尝试链接这些库,但是稍后当有人也链接我的库时这样做;b) 即使我在 SDL 库中进行链接,它仍然找不到它正在查找的 SDL 函数 ( SDL_GetTicks, SDL_Delay) - 错误是相同的。另请注意,一些缺失的项目来自std.

这是错误。如您所见,我正在尝试 ld 上的各种标志,以使其不尝试解析引用,但尚未成功。

C:\Users\...\mcve>make
g++ -c -c -I../../../external/SDL2/include -I../include -o mcve.o mcve.cpp
ld -G --unresolved-symbols=ignore-all --warn-unresolved-symbols  -o libmcve.a mcve.o    
C:\MinGW\bin\ld.exe: mcve.o:mcve.cpp:(.text+0x8): undefined reference to `SDL_GetTicks'
C:\MinGW\bin\ld.exe: mcve.o:mcve.cpp:(.text+0x23): undefined reference to `SDL_GetTicks'
C:\MinGW\bin\ld.exe: mcve.o:mcve.cpp:(.text+0x2f): undefined reference to `SDL_Delay'
C:\MinGW\bin\ld.exe: mcve.o:mcve.cpp:(.text+0x46): undefined reference to `std::ios_base::Init::~Init()'
C:\MinGW\bin\ld.exe: mcve.o:mcve.cpp:(.text+0x67): undefined reference to `std::ios_base::Init::Init()'
C:\MinGW\bin\ld.exe: mcve.o:mcve.cpp:(.text+0x73): undefined reference to `atexit'

这是我的源文件:

#include <SDL.h>
#include <iostream>         //If I take this out, I no longer get the 
                            //unresolved references to std::ios_base::Init::Init,
                            // std::ios_base::Init::~Init, and atexit

Uint32 time;    

void doSomething () 
{
  if (time > SDL_GetTicks ()) 
    SDL_Delay (time - SDL_GetTicks());
}

这是Makefile。如果我取消注释 LDFLAGS 的其余部分并让 SDL 库链接,它不会更改输出。

CFLAGS  =-c -I../../../external/SDL2/include -I../include
LDFLAGS = --unresolved-symbols=ignore-all --warn-unresolved-symbols #-L. -lSDL2 -lSDL2_ttf -lSDL2_image -lSDL2_mixer 

# Files
SOURCE_FILES= mcve.cpp
OBJECT_FILES= mcve.o

libmcve.a: $(OBJECT_FILES)
    ld $(LDFLAGS) -o $@ $^ -G

$(OBJECT_FILES): %.o: $(SOURCE_FILES)
    g++ -c $(CFLAGS) -o $@ $<

标签: makefilestatic-librariesld

解决方案


您正在尝试使用链接器libmcve.a从目标文件 创建静态库。mcve.old

链接器无法生成静态库。静态只是arar.

在 makefile 中创建或更新静态库的方法是:

libmcve.a: $(OBJECT_FILES)
    rm -f $@    # Delete archive if already exists
    ar rcs $@ $^ # Recreate archive with contents $(OBJECT_FILES)

顺便说一句,请注意,您-c在编译命令中传递了两次该选项:

g++ -c -c -I../../../external/SDL2/include -I../include -o mcve.o mcve.cpp

那是因为您已将其包含在您的CFLAGS设置中:

CFLAGS  =-c -I../../../external/SDL2/include -I../include

(不应该在的地方),以及在您的编译配方中:

g++ -c $(CFLAGS) -o $@ $<

(应该在哪里)。


推荐阅读