【问题记录】c++编译报错~/bin/ld.exe: cannot find 0

构建环境

windows 11
cmake 3.29.0-rc2
MinGW 8.1.0

先说结论

推测这应该是MinGW和cmake版本的部分不兼容导致的,尝试使用旧一点的cmake或新一些的MinGW或许会解决这个问题。
进一步进行定位后发现,这是因为project()的位置没对导致的

  • gpt4:

    建立 project 指令通常應該在所有其他命令之前。CMake 推薦在 CMakeLists.txt 文件的開頭,立即在
    cmake_minimum_required 之後。將 project 指令移至全局配置文件中,或者至少在添加子目錄和執行其他設置之前。

所以将project()写到cmake_minimum_required 之后问题解决。
另外如果使用msvc的构建工具时并不会出错。

诡异的 0 是怎么来的

首先这个程序没有链接任何外部库,也没导出任何库;
我通过添加cmake -- VERBOSE=1参数查看详细信息后发现问题出在build.make文件中的这一句命令

F:/MinGW/bin/c++.exe -g \
-Wl,--whole-archive CMakeFiles/print_hello_world.dir/objects.a \
-Wl,--no-whole-archive -o print_hello_world.exe \
-Wl,--out-implib,"" \
-Wl,--major-image-version,0,--minor-image-version,0 @CMakeFiles/print_hello_world.dir/linkLibs.rsp

经过分析问题是在--out-implib,""这个参数,不明白cmake为什么会生成这一句,似乎这个版本的c++.exe没有将""识别为空 ,并且又把--major-image-version后面的主版本号0识别为了一个库,所以才出现了这个链接错误,删除-Wl,--out-implib,""或主版本号0后恢复正常,这里肯定是选择删除-Wl,--out-implib,""才符合语义。
另外如果链接了第三方库,反而不会出问题。

修正前报错~/bin/ld.exe: cannot find 0

F:/MinGW/bin/c++.exe -g \
-Wl,--whole-archive CMakeFiles/print_hello_world.dir/objects.a \
-Wl,--no-whole-archive -o print_hello_world.exe \
-Wl,--out-implib,"" \
-Wl,--major-image-version,0,--minor-image-version,0 @CMakeFiles/print_hello_world.dir/linkLibs.rsp

修正后正常

F:/MinGW/bin/c++.exe -g \
-Wl,--whole-archive CMakeFiles/print_hello_world.dir/objects.a \
-Wl,--no-whole-archive -o print_hello_world.exe \
-Wl,--major-image-version,0,--minor-image-version,0 @CMakeFiles/print_hello_world.dir/linkLibs.rsp

链接gtest静态库后

源码结构没有变化,但该行参数变成了-Wl,--out-implib,libtest_print_hello_world.dll.a

F:/MinGW/bin/c++.exe -g \
-Wl,--whole-archive CMakeFiles/test_print_hello_world.dir/objects.a \
-Wl,--no-whole-archive -o test_print_hello_world.exe \
-Wl,--out-implib,libtest_print_hello_world.dll.a \
-Wl,--major-image-version,0,--minor-image-version,0 @CMakeFiles/test_print_hello_world.dir/linkLibs.rsp

gtp4给出的解释是这样的:

连接器(linker)被指令以特定的方式处理静态库和生成的输出。这不仅仅关于链接gtest.a。让我们逐一分析参数:

  1. -Wl,--whole-archive CMakeFiles/test_print_hello_world.dir/objects.a -Wl,--no-whole-archive: 这一段指示连接器将指定的静态库(objects.a)中所有的目标文件全部加入到最终的可执行文件中。--whole-archive 选项强制链接器不忽略任何目标文件,哪怕它们看起来并没有被使用。这通常用来确保库中可能会被程序运行时动态引用的对象都将被包含进来。
  2. -o test_print_hello_world.exe: 这指示生成的输出是一个名为 test_print_hello_world.exe 的可执行文件。
  3. -Wl,--out-implib,libtest_print_hello_world.dll.a: 这段指示连接器创建一个导入库(import library),名字叫做 libtest_print_hello_world.dll.a。在Windows平台,当你创建一个DLL的时候,你通常也会生成一个关联的导入库,它用于从DLL中链接函数。即使你的意图是创建一个可执行文件,连接器也可以生成这样一个导入库,它可以被其他的DLL或可执行文件使用,来链接到你的程序(尽管这在实际中并不常见)。

后缀.dll.a是在MinGW或Cygwin环境中使用的,它表示这是一个用于链接DLL的导入库。与.lib文件相比(通常在Native Windows环境中用作导入库),.dll.a 的格式略有不同,以符合GCC工具链的预期。

如果你的目标确实是生成一个独立的可执行文件,你可能不需要 --out-implib 参数,除非你打算将你的可执行程序输出一些接口给其他程序动态链接使用。只有在生成DLL时,--out-implib 通常才是必要的。

如果有进一步的混淆,你可以检查CMakeList.txt中的设置,或者是和其他库或目标的交互方式,来确定为什么这个命令行被这样配置。这可能需要你具体了解项目的构建系统配置。

  • 6
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
这个错误提示表明链接器无法找到名为 libxiffs.a 的库文件。在编译链接 C/C++ 代码时,链接器需要找到所需的库文件才能生成可执行文件。 要解决这个问题,你需要检查以下几点: 1. 确认库文件存在。在你的项目中,确保 libxiffs.a 库文件已经存在,并且位于正确的路径下。如果库文件不存在或路径不正确,链接器自然就无法找到它。 2. 确认库文件名正确。链接器通常会使用 -l 选项和库文件名来指定需要链接的库文件。在你的编译命令或 Makefile 中,检查 -l 选项后面的库文件名是否正确。在这个错误提示中,-llibxiffs.a 表示链接器会寻找名为 libxiffs.a 的库文件。如果库文件名不正确,链接器也会找不到它。 3. 确认库文件路径正确。如果库文件位于非标准路径下,你需要在编译命令或 Makefile 中使用 -L 选项来指定库文件的路径。例如,如果库文件位于 /usr/local/lib 目录下,你需要在编译命令或 Makefile 中添加 -L/usr/local/lib 选项。 4. 确认库文件可读。在你的项目中,确保库文件具有可读权限。如果文件权限不正确,链接器可能无法读取库文件。 如果你已经检查过以上几点,仍然无法解决问题,你可以尝试在编译命令或 Makefile 中手动指定库文件的路径和名称,例如: ``` gcc main.c -o main -L/path/to/lib -lxiffs ``` 其中 /path/to/lib 是库文件所在的路径,-lxiffs 表示链接器需要链接名为 libxiffs.a 的库文件。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值