c++ - 如何在 clang 中重新启用 C++17 已删除的功能?
问题描述
我有一些作为大型项目的一部分构建的 3rdparty 库。一些库需要 C++17,其他库需要 C++17 ( std::auto_ptr
, std::fun_ptr
) 中删除的功能。
对于这个项目,使用了 Clang(版本 11)。如果我使用 clang 定义,通过搜索所需版本的不匹配应该不是问题_LIBCPP_ENABLE_CXX17_REMOVED_FEATURES
。
但是,这具有零影响。我在一个很小的测试样本中尝试了这个定义:
#include <memory>
int main(void)
{
std::auto_ptr<int> thing(new int(1));
return 0;
}
不指定 C++ std
,它编译得很好。
但是,如果我指定 C++17,它会失败-D_LIBCPP_ENABLE_CXX17_REMOVED_FEATURES
PS E:\> clang++ -std=c++17 -D_LIBCPP_ENABLE_CXX17_REMOVED_FEATURES .\autoptr.cpp
.\autoptr.cpp:5:10: error: no member named 'auto_ptr' in namespace 'std'
std::auto_ptr<int> thing(new int(1));
~~~~~^
.\autoptr.cpp:5:22: error: expected '(' for function-style cast or type construction
std::auto_ptr<int> thing(new int(1));
~~~^
.\autoptr.cpp:5:24: error: use of undeclared identifier 'thing'
std::auto_ptr<int> thing(new int(1));
^
3 errors generated.
C++ 17 删除的功能在 Clang 中不再可用,还是我需要额外的编译器选项?
叮当版本:
PS E:\> clang++ -v
clang version 11.1.0
Target: x86_64-pc-windows-msvc
Thread model: posix
InstalledDir: C:\Program Files\LLVM\bin
编辑:有趣,这在我拥有的不同系统上编译得很好。无需额外定义。
#@#:~/kraken$ clang++ -v
clang version 10.0.0-4ubuntu1~18.04.2
Target: x86_64-pc-linux-gnu
Thread model: posix
InstalledDir: /usr/bin
edit2:从 11.1.0 升级到 12.0.1。没变
$ clang++ -v -stdlib=libc++ -std=c++17 -D_LIBCPP_ENABLE_CXX17_REMOVED_FEATURES -D_LIBCPP_ENABLE_CXX17_REMOVED_AUTO_PTR autoptr.cpp
clang version 12.0.1
Target: x86_64-pc-windows-msvc
Thread model: posix
InstalledDir: C:\Program Files\LLVM\bin
clang++: warning: argument unused during compilation: '-stdlib=libc++' [-Wunused-command-line-argument]
"C:\\Program Files\\LLVM\\bin\\clang++.exe" -cc1 -triple x86_64-pc-windows-msvc19.29.30038 -emit-obj -mrelax-all -mincremental-linker-compatible --mrelax-relocations -disable-free -disable-llvm-verifier -discard-value-names -main-file-name autoptr.cpp -mrelocation-model pic -pic-level 2 -mframe-pointer=none -fmath-errno -fno-rounding-math -mconstructor-aliases -munwind-tables -target-cpu x86-64 -tune-cpu generic -v -resource-dir "C:\\Program Files\\LLVM\\lib\\clang\\12.0.1" -D _LIBCPP_ENABLE_CXX17_REMOVED_FEATURES -D _LIBCPP_ENABLE_CXX17_REMOVED_AUTO_PTR -internal-isystem "C:\\Program Files\\LLVM\\lib\\clang\\12.0.1\\include" -internal-isystem "C:\\Program Files (x86)\\Microsoft Visual Studio\\2019\\Professional\\VC\\Tools\\MSVC\\14.29.30037\\include" -internal-isystem "C:\\Program Files (x86)\\Microsoft Visual Studio\\2019\\Professional\\VC\\Tools\\MSVC\\14.29.30037\\atlmfc\\include" -internal-isystem "C:\\Program Files (x86)\\Windows Kits\\10\\Include\\10.0.19041.0\\ucrt" -internal-isystem "C:\\Program Files (x86)\\Windows Kits\\10\\include\\10.0.19041.0\\shared" -internal-isystem "C:\\Program Files (x86)\\Windows Kits\\10\\include\\10.0.19041.0\\um" -internal-isystem "C:\\Program Files (x86)\\Windows Kits\\10\\include\\10.0.19041.0\\winrt" -std=c++17 -fdeprecated-macro -fdebug-compilation-dir "E:\\" -ferror-limit 19 -fno-use-cxa-atexit -fms-extensions -fms-compatibility -fms-compatibility-version=19.29.30038 -fdelayed-template-parsing -fcxx-exceptions -fexceptions -faddrsig -o "C:\\Users\\JONATH~1.NOB\\AppData\\Local\\Temp\\autoptr-152d67.o" -x c++ autoptr.cpp
clang -cc1 version 12.0.1 based upon LLVM 12.0.1 default target x86_64-pc-windows-msvc
#include "..." search starts here:
#include <...> search starts here:
C:\Program Files\LLVM\lib\clang\12.0.1\include
C:\Program Files (x86)\Microsoft Visual Studio\2019\Professional\VC\Tools\MSVC\14.29.30037\include
C:\Program Files (x86)\Microsoft Visual Studio\2019\Professional\VC\Tools\MSVC\14.29.30037\atlmfc\include
C:\Program Files (x86)\Windows Kits\10\Include\10.0.19041.0\ucrt
C:\Program Files (x86)\Windows Kits\10\include\10.0.19041.0\shared
C:\Program Files (x86)\Windows Kits\10\include\10.0.19041.0\um
C:\Program Files (x86)\Windows Kits\10\include\10.0.19041.0\winrt
End of search list.
autoptr.cpp:5:10: error: no member named 'auto_ptr' in namespace 'std'
std::auto_ptr<int> thing(new int(1));
~~~~~^
autoptr.cpp:5:22: error: expected '(' for function-style cast or type construction
std::auto_ptr<int> thing(new int(1));
~~~^
autoptr.cpp:5:24: error: use of undeclared identifier 'thing'
std::auto_ptr<int> thing(new int(1));
^
3 errors generated.
此输出显示了问题。未使用 LLVM libcxx。使用 -D_HAS_AUTO_PTR_ETC=1 为我解决了这个问题。
解决方案
使用 clang 的 -v 标志检查哪个 c++ stdlib 正在使用。我假设正在使用 llvm 的 libcxx,因为 -stdlib=libcxx 被忽略了。事实上,msvc 的 stdlib 正在被使用。与 msvc 一起使用:
clang++ -std=c++17 -D_HAS_AUTO_PTR_ETC=1 file.cpp
推荐阅读
- javascript - 我应该将谷歌应用脚本函数创建为 const
- angular - 400 Bad request CSRF Angular 8 with no razor pages
- python - 是否有注册模块导入/加载挂钩的机制?
- python - 在两个单词之间查找文本并替换它的最有效/pythonic 方式
- spring-boot - 对于 Azure servicebus jms spring boot starter,如果处理失败,是否有重新排队消息的属性设置?
- node.js - 制作坚如磐石的电子邮件以在 nodejs 应用程序内进行大量发送的方法?
- flutter - Flutter Gradle 版本问题
- python - 将文件夹名称输出到数据框中的列?
- java - Apache PDFBox PDPageContentStream showText 和 drawImage 有时不起作用或显示颠倒
- bazel - 如何在工作区*内指定 http_archive 的凭据?