experimental :: filesystem链接器错误

时间:2015-10-15 13:28:28

标签: c++ gcc c++17

我尝试在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

有关如何链接新c ++ 1z功能的任何提示?

6 个答案:

答案 0 :(得分:108)

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

您只需要与-lstdc++fs关联即可使用它。

(相关库libstdc++fs.a是一个静态库,因此对于任何静态库,它应该在链接器命令中依赖它的任何对象之后<)>。

2017年11月更新以及文件系统TS,现在GCC主干有一个C ++ 17文件系统库的实现,在{{1}中定义在使用<filesystem>std::filesystem时,在名称空间-std=gnu++17中(注意不是&#34;实验&#34;在这些名称中)。 GCC的C ++ 17支持还不完整或稳定,并且在它被认为准备好黄金时间使用之前,您还需要链接到-std=c++17以获取C ++ 17文件系统功能。

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

答案 1 :(得分:12)

使用clang 4.0+,您需要链接libc++experimental.a

确保使用-stdlib = libc ++(如评论中所述)使用libc ++(而不是libstdc ++)构建

答案 2 :(得分:9)

如果您使用的是cmake,请将以下行添加到CMakeLists.txt

link_libraries(stdc++fs)

以便cmake可以链接到相应的库。

答案 3 :(得分:1)

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

env:el6gcc/5.5.0

#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

以下内容存在编译错误_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

答案 4 :(得分:1)

对于

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")

这不是用于Apple Clang,而是用于从官方https://releases.llvm.org或程序包管理器安装的LLVM clang。

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

答案 5 :(得分:-1)

You can easily try my code online.

//  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"