如何确定和更改 g++ 正在使用的 C++ 标准库的版本?

如何确定和更改 g++ 正在使用的 C++ 标准库的版本?

我有一个 C++ 程序,我正在尝试更新它以使用新 C++-17 标准库中的 std::filesystem 库。我必须安装 g++ 版本 8 才能获得 C++-17 功能。我的程序编译了,但无法链接,我怀疑这是因为我安装了多个版本的 C++ 标准库,一个用于 g++ 版本 7,一个用于 g++ 版本 8。我需要它与 libstdc++-8-dev 链接才能获得 C++-17 功能。

我如何确定 g++ 正在使用哪个版本的 libstdc++?是否可以切换版本?理想情况下,我确实希望我的计算机上有多个版本,因为一些 Ubuntu 软件包编译 C++ 代码并且可能需要旧版本,但我希望我的程序使用 g++ 8,而 Ubuntu 使用它通常使用的版本。

更新:下面描述的我遇到的错误已解决(见评论)。但我仍然对上面提出的问题感到好奇。

我得到的错误与这些类似: https://stackoverflow.com/questions/51252829/stdfilesystemdirectory-iterator-linker-issue-c17

但是我尝试了他们的解决方案,添加了 -lstdc++fs 选项,但没有什么区别。

编辑:我找到了 std::filesystem 的静态库文件,并通过将其添加到 g++ 链接命令中明确链接它,错误就消失了:

g++ $(CXXFLAGS) $(obj) /usr/lib/gcc/x86_64-linux-gnu/8/libstdc++fs.a

这说明在链接时确实找不到正确版本的标准库。这是为什么呢?

编辑:根据以下要求,这是结果locate libstdc

/snap/core/5328/usr/lib/x86_64-linux-gnu/libstdc++.so.6  
/snap/core/5328/usr/lib/x86_64-linux-gnu/libstdc++.so.6.0.21  
/snap/core/5328/usr/share/doc/libstdc++6  
/snap/core/5328/usr/share/gcc-5/python/libstdcxx  
/snap/core/5328/usr/share/gcc-5/python/libstdcxx/__init__.py  
/snap/core/5328/usr/share/gcc-5/python/libstdcxx/v6  
/snap/core/5328/usr/share/gcc-5/python/libstdcxx/v6/__init__.py  
/snap/core/5328/usr/share/gcc-5/python/libstdcxx/v6/printers.py  
/snap/core/5328/usr/share/gcc-5/python/libstdcxx/v6/xmethods.py  
/snap/core/5328/usr/share/gdb/auto-load/usr/lib/x86_64-linux-gnu/libstdc++.so.6.0.21-gdb.py  
/usr/lib/gcc/x86_64-linux-gnu/7/libstdc++.a  
/usr/lib/gcc/x86_64-linux-gnu/7/libstdc++.so  
/usr/lib/gcc/x86_64-linux-gnu/7/libstdc++fs.a  
/usr/lib/gcc/x86_64-linux-gnu/8/libstdc++.a  
/usr/lib/gcc/x86_64-linux-gnu/8/libstdc++.so  
/usr/lib/gcc/x86_64-linux-gnu/8/libstdc++fs.a  
/usr/lib/i386-linux-gnu/libstdc++.so.6  
/usr/lib/i386-linux-gnu/libstdc++.so.6.0.25  
/usr/lib/x86_64-linux-gnu/libstdc++.so.6  
/usr/lib/x86_64-linux-gnu/libstdc++.so.6.0.25  
/usr/lib32/libstdc++.so.6  
/usr/lib32/libstdc++.so.6.0.25  
/usr/share/doc/libstdc++-7-dev  
/usr/share/doc/libstdc++-8-dev  
/usr/share/doc/libstdc++6  
/usr/share/doc/gcc-7-base/C++/README.libstdc++-baseline.amd64  
/usr/share/doc/gcc-7-base/C++/changelog.libstdc++.gz  
/usr/share/doc/gcc-7-base/C++/libstdc++_symbols.txt.amd64  
/usr/share/doc/gcc-8-base/C++/README.libstdc++-baseline.amd64  
/usr/share/doc/gcc-8-base/C++/changelog.libstdc++.gz  
/usr/share/doc/gcc-8-base/C++/libstdc++_symbols.txt.amd64  
/usr/share/gcc-8/python/libstdcxx  
/usr/share/gcc-8/python/libstdcxx/__init__.py  
/usr/share/gcc-8/python/libstdcxx/v6  
/usr/share/gcc-8/python/libstdcxx/v6/__init__.py  
/usr/share/gcc-8/python/libstdcxx/v6/printers.py  
/usr/share/gcc-8/python/libstdcxx/v6/xmethods.py  
/usr/share/gdb/auto-load/usr/lib/i386-linux-gnu/libstdc++.so.6.0.25-gdb.py  
/usr/share/gdb/auto-load/usr/lib/x86_64-linux-gnu/libstdc++.so.6.0.25-gdb.py  
/usr/share/gdb/auto-load/usr/lib32/libstdc++.so.6.0.25-gdb.py  
/usr/share/locale-langpack/en_GB/LC_MESSAGES/libstdc++.mo  
/var/cache/apt/archives/libstdc++-7-dev_7.3.0-16ubuntu3_amd64.deb  
/var/cache/apt/archives/libstdc++6_8-20180414-1ubuntu2_i386.deb  
/var/lib/dpkg/info/libstdc++-7-dev:amd64.list  
/var/lib/dpkg/info/libstdc++-7-dev:amd64.md5sums  
/var/lib/dpkg/info/libstdc++-8-dev:amd64.list  
/var/lib/dpkg/info/libstdc++-8-dev:amd64.md5sums  
/var/lib/dpkg/info/libstdc++6:amd64.list  
/var/lib/dpkg/info/libstdc++6:amd64.md5sums  
/var/lib/dpkg/info/libstdc++6:amd64.postinst  
/var/lib/dpkg/info/libstdc++6:amd64.prerm  
/var/lib/dpkg/info/libstdc++6:amd64.shlibs  
/var/lib/dpkg/info/libstdc++6:amd64.symbols  
/var/lib/dpkg/info/libstdc++6:amd64.triggers  
/var/lib/dpkg/info/libstdc++6:i386.list  
/var/lib/dpkg/info/libstdc++6:i386.md5sums  
/var/lib/dpkg/info/libstdc++6:i386.postinst  
/var/lib/dpkg/info/libstdc++6:i386.prerm  
/var/lib/dpkg/info/libstdc++6:i386.shlibs  
/var/lib/dpkg/info/libstdc++6:i386.symbols  
/var/lib/dpkg/info/libstdc++6:i386.triggers  

编辑:这是一个简单的示例程序,展示了我的机器上的问题:

#include <filesystem>
#include <iostream>

int main()
{
    std::filesystem::path p = "/";
    std::cout << p;
}

编辑:我编译了示例程序

g++ -v -Wall -std=c++1z test.cpp

并得到以下输出:

Using built-in specs.
COLLECT_GCC=g++
COLLECT_LTO_WRAPPER=/usr/lib/gcc/x86_64-linux-gnu/8/lto-wrapper
OFFLOAD_TARGET_NAMES=nvptx-none
OFFLOAD_TARGET_DEFAULT=1
Target: x86_64-linux-gnu
Configured with: ../src/configure -v --with-pkgversion='Ubuntu 8-20180414-1ubuntu2' --with-bugurl=file:///usr/share/doc/gcc-8/README.Bugs --enable-languages=c,ada,c++,go,brig,d,fortran,objc,obj-c++ --prefix=/usr --with-gcc-major-version-only --with-as=/usr/bin/x86_64-linux-gnu-as --with-ld=/usr/bin/x86_64-linux-gnu-ld --program-suffix=-8 --program-prefix=x86_64-linux-gnu- --enable-shared --enable-linker-build-id --libexecdir=/usr/lib --without-included-gettext --enable-threads=posix --libdir=/usr/lib --enable-nls --with-sysroot=/ --enable-clocale=gnu --enable-libstdcxx-debug --enable-libstdcxx-time=yes --with-default-libstdcxx-abi=new --enable-gnu-unique-object --disable-vtable-verify --enable-libmpx --enable-plugin --enable-default-pie --with-system-zlib --with-target-system-zlib --enable-objc-gc=auto --enable-multiarch --disable-werror --with-arch-32=i686 --with-abi=m64 --with-multilib-list=m32,m64,mx32 --enable-multilib --with-tune=generic --enable-offload-targets=nvptx-none --without-cuda-driver --enable-checking=release --build=x86_64-linux-gnu --host=x86_64-linux-gnu --target=x86_64-linux-gnu
Thread model: posix
gcc version 8.0.1 20180414 (experimental) [trunk revision 259383] (Ubuntu 8-20180414-1ubuntu2) 
COLLECT_GCC_OPTIONS='-v' '-Wall' '-std=c++17' '-shared-libgcc' '-mtune=generic' '-march=x86-64'
 /usr/lib/gcc/x86_64-linux-gnu/8/cc1plus -quiet -v -imultiarch x86_64-linux-gnu -D_GNU_SOURCE test.cpp -quiet -dumpbase test.cpp -mtune=generic -march=x86-64 -auxbase test -Wall -std=c++17 -version -fstack-protector-strong -Wformat-security -o /tmp/cchNMFhq.s
GNU C++17 (Ubuntu 8-20180414-1ubuntu2) version 8.0.1 20180414 (experimental) [trunk revision 259383] (x86_64-linux-gnu)
    compiled by GNU C version 8.0.1 20180414 (experimental) [trunk revision 259383], GMP version 6.1.2, MPFR version 4.0.1, MPC version 1.1.0, isl version isl-0.19-GMP

GGC heuristics: --param ggc-min-expand=100 --param ggc-min-heapsize=131072
ignoring duplicate directory "/usr/include/x86_64-linux-gnu/c++/8"
ignoring nonexistent directory "/usr/local/include/x86_64-linux-gnu"
ignoring nonexistent directory "/usr/lib/gcc/x86_64-linux-gnu/8/../../../../x86_64-linux-gnu/include"
#include "..." search starts here:
#include <...> search starts here:
 /usr/include/c++/8
 /usr/include/x86_64-linux-gnu/c++/8
 /usr/include/c++/8/backward
 /usr/lib/gcc/x86_64-linux-gnu/8/include
 /usr/local/include
 /usr/lib/gcc/x86_64-linux-gnu/8/include-fixed
 /usr/include/x86_64-linux-gnu
 /usr/include
End of search list.
GNU C++17 (Ubuntu 8-20180414-1ubuntu2) version 8.0.1 20180414 (experimental) [trunk revision 259383] (x86_64-linux-gnu)
    compiled by GNU C version 8.0.1 20180414 (experimental) [trunk revision 259383], GMP version 6.1.2, MPFR version 4.0.1, MPC version 1.1.0, isl version isl-0.19-GMP

GGC heuristics: --param ggc-min-expand=100 --param ggc-min-heapsize=131072
Compiler executable checksum: 3e256d94cf1a8afbef1b441ddb5fe0dc
COLLECT_GCC_OPTIONS='-v' '-Wall' '-std=c++17' '-shared-libgcc' '-mtune=generic' '-march=x86-64'
 /usr/bin/x86_64-linux-gnu-as -v --64 -o /tmp/cc4KG8eJ.o /tmp/cchNMFhq.s
GNU assembler version 2.30 (x86_64-linux-gnu) using BFD version (GNU Binutils for Ubuntu) 2.30
COMPILER_PATH=/usr/lib/gcc/x86_64-linux-gnu/8/:/usr/lib/gcc/x86_64-linux-gnu/8/:/usr/lib/gcc/x86_64-linux-gnu/:/usr/lib/gcc/x86_64-linux-gnu/8/:/usr/lib/gcc/x86_64-linux-gnu/
LIBRARY_PATH=/usr/lib/gcc/x86_64-linux-gnu/8/:/usr/lib/gcc/x86_64-linux-gnu/8/../../../x86_64-linux-gnu/:/usr/lib/gcc/x86_64-linux-gnu/8/../../../../lib/:/lib/x86_64-linux-gnu/:/lib/../lib/:/usr/lib/x86_64-linux-gnu/:/usr/lib/../lib/:/usr/lib/gcc/x86_64-linux-gnu/8/../../../:/lib/:/usr/lib/
COLLECT_GCC_OPTIONS='-v' '-Wall' '-std=c++17' '-shared-libgcc' '-mtune=generic' '-march=x86-64'
 /usr/lib/gcc/x86_64-linux-gnu/8/collect2 -plugin /usr/lib/gcc/x86_64-linux-gnu/8/liblto_plugin.so -plugin-opt=/usr/lib/gcc/x86_64-linux-gnu/8/lto-wrapper -plugin-opt=-fresolution=/tmp/ccoQuLk2.res -plugin-opt=-pass-through=-lgcc_s -plugin-opt=-pass-through=-lgcc -plugin-opt=-pass-through=-lc -plugin-opt=-pass-through=-lgcc_s -plugin-opt=-pass-through=-lgcc --sysroot=/ --build-id --eh-frame-hdr -m elf_x86_64 --hash-style=gnu --as-needed -dynamic-linker /lib64/ld-linux-x86-64.so.2 -pie -z now -z relro /usr/lib/gcc/x86_64-linux-gnu/8/../../../x86_64-linux-gnu/Scrt1.o /usr/lib/gcc/x86_64-linux-gnu/8/../../../x86_64-linux-gnu/crti.o /usr/lib/gcc/x86_64-linux-gnu/8/crtbeginS.o -L/usr/lib/gcc/x86_64-linux-gnu/8 -L/usr/lib/gcc/x86_64-linux-gnu/8/../../../x86_64-linux-gnu -L/usr/lib/gcc/x86_64-linux-gnu/8/../../../../lib -L/lib/x86_64-linux-gnu -L/lib/../lib -L/usr/lib/x86_64-linux-gnu -L/usr/lib/../lib -L/usr/lib/gcc/x86_64-linux-gnu/8/../../.. /tmp/cc4KG8eJ.o -lstdc++ -lm -lgcc_s -lgcc -lc -lgcc_s -lgcc /usr/lib/gcc/x86_64-linux-gnu/8/crtendS.o /usr/lib/gcc/x86_64-linux-gnu/8/../../../x86_64-linux-gnu/crtn.o
/tmp/cc4KG8eJ.o: In function `std::filesystem::__cxx11::path::path<char [2], std::filesystem::__cxx11::path>(char const (&) [2], std::filesystem::__cxx11::path::format)':
test.cpp:(.text._ZNSt10filesystem7__cxx114pathC2IA2_cS1_EERKT_NS1_6formatE[_ZNSt10filesystem7__cxx114pathC5IA2_cS1_EERKT_NS1_6formatE]+0x6d): undefined reference to `std::filesystem::__cxx11::path::_M_split_cmpts()'
collect2: error: ld returned 1 exit status

答案1

我的程序可以编译但无法链接,我怀疑这是因为我安装了多个版本的 C++ 标准库,一个用于 g++ 版本 7,一个用于 g++ 版本 8。我需要它与 libstdc++-8-dev 链接以获取 C++-17 功能。

程序无法链接的原因是您在编译过程中没有传递正确的参数。编译器可以看到函数声明,但链接器找不到函数实现。

您需要通过明确告知要使用的库版本来告诉系统要使用哪个版本/实现,您可以使用参数来执行此操作-l。 在您的例子中,您可能需要-l在参数前面加上参数-L

以下是一个例子:

g++ myfile.cpp -L/usr/lib/gcc/x86_64-linux-gnu/7 -lstdc++

如果您已经安装了 v8,除非库安装在其他地方,否则您可以执行以下操作:

g++ myfile.cpp -L/usr/lib/gcc/x86_64-linux-gnu/8 -lstdc++

那就可以了。

请执行下列操作:

sudo updatedb
locate libstdc

并编辑您的问题并发布输出。我将尝试构建精确的命令。

更新:我可以看到你有:

/usr/lib/gcc/x86_64-linux-gnu/8/libstdc++.a  
/usr/lib/gcc/x86_64-linux-gnu/8/libstdc++.so  
/usr/lib/gcc/x86_64-linux-gnu/8/libstdc++fs.a  

你能看看这是否有效吗?

g++ test.cpp -std=c++11 -L/usr/lib/gcc/x86_64-linux-gnu/8 -lstdc++

相关内容