文件系统链接器错误

我尝试在 gcc 6.0的开发中使用新的 c + + 1z 特性。

如果我试试这个小例子:

#include <iostream>
#include <experimental/filesystem>
namespace fs = std::experimental::filesystem;
int main()
{
fs::path p1 = "/home/pete/checkit";


std::cout << "p1 = " << p1 << std::endl;
}

我有:

/opt/linux-gnu_6-20151011/bin/g++ --std=c++1z main.cpp -O2 -g -o go
/tmp/ccaGzqFO.o: In function \`std::experimental::filesystem::v1::__cxx11::path::path(char const (&) [36])':
/opt/linux-gnu_6-20151011/include/c++/6.0.0/experimental/bits/fs_path.h:167: undefined reference to `std::experimental::filesystem::v1::__cxx11::path::_M_split_cmpts()'
collect2: error: ld returned 1 exit status

Gcc 版本是快照 linux-gnu _ 6-20151011

Any hints how to link for the new c++1z features?

89683 次浏览

文件系统 TS 与 C + + 1z 支持没有任何关系,它是一个完全独立的规范,不是 C + + 1z 工作草案的一部分。GCC 的实现(在 GCC 5.3及更高版本中)甚至可以在 C + + 11模式下使用。

您只需要与 -lstdc++fs链接就可以使用它。

(The relevant library, libstdc++fs.a, is a static library, so as with any static library it should come 之后 any objects that depend on it in the linker command.)

更新于2017年11月: 和文件系统 TS 一样,GCC 8.x 还有有一个 C + + 17文件系统库的实现,在使用 -std=gnu++17-std=c++17时,在 <filesystem>和名称空间 std::filesystem(注意,这些名称中没有“实验”)中定义。GCC 的 C + + 17支持尚不完整或稳定,在它被认为已经准备好在黄金时间使用之前,您还需要链接到 -lstdc++fs以获得 C + + 17文件系统的特性。

2019年1月更新: 从 GCC 9开始,C + + 17 std::filesystem组件可以在没有 -lstdc++fs的情况下使用(但是 std::experimental::filesystem仍然需要该库)。

对于 clang 4.0 + ,您需要与 libc++experimental.a进行链接

Make sure you're building with libc++ (not libstdc++) with the -stdlib=libc++ (as mentioned in the comments)

下面是一个可能对将来的某人有帮助的演示:

编译: pestwave

#include <iostream>
#include <string>
#include <experimental/filesystem>


int main()
{
std::string path = std::experimental::filesystem::current_path();


std::cout << "path = " << path << std::endl;
}

下面是编译和测试过程,标志是 -std=c++17 -lstdc++fs:

$ g++ -v
Using built-in specs.
COLLECT_GCC=g++
COLLECT_LTO_WRAPPER=/apps/gcc-5.5.0/bin/../libexec/gcc/x86_64-unknown-linux-gnu/5.5.0/lto-wrapper
Target: x86_64-unknown-linux-gnu
Configured with: ../configure --prefix=/apps/gcc-5.5.0 --disable-multilib --enable-shared --enable-threads=posix --enable-__cxa_atexit --enable-clocale=gnu --enable-languages=all
Thread model: posix
gcc version 5.5.0 (GCC)


$ ls -lrt /apps/gcc-5.5.0/lib64 | grep libstdc
-rwxr-xr-x. 1 root root  11272436 Jun 25 10:51 libstdc++.so.6.0.21
-rw-r--r--. 1 root root      2419 Jun 25 10:51 libstdc++.so.6.0.21-gdb.py
-rwxr-xr-x. 1 root root       976 Jun 25 10:51 libstdc++.la
-rwxr-xr-x. 1 root root  11272436 Jun 25 10:51 libstdc++.so
-rw-r--r--. 1 root root  10581732 Jun 25 10:51 libstdc++fs.a
-rw-r--r--. 1 root root  28985412 Jun 25 10:51 libstdc++.a
-rwxr-xr-x. 1 root root       916 Jun 25 10:51 libstdc++fs.la
-rwxr-xr-x. 1 root root  11272436 Jun 25 10:51 libstdc++.so.6


$ g++ filesystem-testing.cpp -lstdc++fs -std=c++17
$ ./a.out


$ g++ -std=c++17 filesystem-testing.cpp -lstdc++fs
$ ./a.out
path = /home/userid/projects-c++/filesystem-testing

它也适用于标志: -std=c++11

$ g++ -std=c++11 filesystem-testing.cpp -lstdc++fs
$ ./a.out
path = /home/userid/projects-c++/filesystem-testing

The follows had compiling error _ZNSt12experimental10filesystem2v112current_pathB5cxx11Ev

$ g++ -std=c++17 -lstdc++fs filesystem-testing.cpp
/tmp/ccA6Q9oF.o: In function `main':
filesystem-testing.cpp:(.text+0x11): undefined reference to `_ZNSt12experimental10filesystem2v112current_pathB5cxx11Ev'
collect2: error: ld returned 1 exit status

Extra Notes:

下面的链接可能会有帮助

如何使用 devtoolset-8-gcc 安装 gcc8

如果您正在使用 cmake,请将以下代码行添加到 CMakeLists.txt:

link_libraries(stdc++fs)

这样 cmake 就可以链接到相应的库。

为了

dyld: lazy symbol binding failed: Symbol not found:
__ZNSt3__14__fs10filesystem4path17replace_extensionERKS2_

还有

Undefined symbols for architecture x86_64:
"std::__1::__fs::filesystem::__current_path(std::__1::error_code*)",
referenced from:
sys::path::get_cwd() in Path.cc.o
ld: symbol(s) not found for architecture x86_64

. . 尝试以下方法:

对于 LLVM clang > = 10,链接 LLVM 提供的 libc++.1.0.dylib

add_link_options("-Wl,-rpath,location_of_llvm_install/lib;location_of_llvm_install/lib/libc++.1.0.dylib")

这不是为苹果叮当声,而是为 LLVM 叮当声安装从官方 https://releases.llvm.org或由一个包管理器。

Xcode < 11没有文件系统头。MacOS < 10.15在系统 dylib 中的 /usr/lib/libc++.1.0.dylib没有 std::filesystem::path符号

你可以很容易地在网上尝试我的代码。

//  currentPath.cpp
// https://stackoverflow.com/questions/33149878#65128193
#include <experimental/filesystem>
#include <iostream>
using namespace std;


int main() {
cout << "path = " << experimental::filesystem::current_path() << endl;
}

编译并运行:

clang++ currentPath.cpp -lstdc++fs && ./a.out # Linux
clang++ currentPath.cpp -lstdc++fs && ./a.exe # MSYS2, Windows

注意: -lstdc++fs是一个链接器标志,而不是编译器标志。 (编写 makefile 时非常重要。)

预期产出:

path = "/path/to/the/current/directory"