C++17
来自 cppreference.cn
< cpp
C++17 是一个主要版本,它紧随次要版本 C++14 之后,并引入了新的语言和库特性。该标准于 2017 年 12 月发布。
以下特性已合并到 C++17 中:
- 来自 文件系统 TS:文件系统库。
- 来自 库基础 v1 TS:包括 std::any、std::optional、std::string_view、std::apply、多态分配器、搜索器 等特性。
- 来自 库基础 v2 TS:std::void_t、std::conjunction、std::disjunction、std::negation、std::not_fn、std::gcd、std::lcm。
- 来自 并行性 v1 TS:包括 执行策略、std::reduce、std::inclusive_scan、std::exclusive_scan 等特性,但移除了
exception_list
。 - 来自 数学特殊函数 IS:数学特殊函数。
- 来自 C11:std::aligned_alloc、std::timespec_get。
本节不完整 |
目录 |
[编辑] 过时特性
[编辑] 移除特性
- std::auto_ptr,
- 弃用的函数对象,
- std::random_shuffle,
- std::unexpected,
- 过时的
iostream
别名, - 三字母词,
- register 关键字,
-
bool
增量, - 动态异常规范
[编辑] 弃用特性
- std::iterator,
- std::raw_storage_iterator,
- std::get_temporary_buffer,
- std::is_literal_type,
- std::result_of,
- 所有 <codecvt>
[编辑] 新语言特性
-
u8
字符字面量 - 将 noexcept 作为类型系统的一部分
- 新的 求值顺序 规则
- lambda 对 *this 的捕获
- 命名空间
- 简化的嵌套命名空间
- 声明多个名称的
using
-声明 - 属性命名空间 无需重复
- 新 属性:
-
[[fallthrough]]
-
[[maybe_unused]]
-
[[nodiscard]]
-
- __has_include
[编辑] 新头文件
[编辑] 新库特性
[编辑] 工具类型
[编辑] 内存管理
- 未初始化内存算法
- weak_from_this
- std::pmr::memory_resource 和 std::polymorphic_allocator
- std::aligned_alloc
- 透明 std::owner_less
- 对 std::shared_ptr 的数组支持
- 具有显式对齐的 分配函数
[编辑] 编译时编程
- std::byte
- std::conjunction/std::disjunction/std::negation
- 类型特性 变量模板 (
xxx_+v
) - std::is_swappable
- std::is_invocable
- std::is_aggregate
- std::has_unique_object_representations
[编辑] 算法
[编辑] 迭代器和容器
- map/set extract 和 map/set merge
- map/unordered_map try_emplace 和 insert_or_assign
- 连续迭代器 (LegacyContiguousIterator)
- 非成员 std::size/std::empty/std::data
[编辑] 数值
- 数学特殊函数
- 3D std::hypot
[编辑] 其他
- 缓存行接口
- std::launder
- std::uncaught_exceptions
- std::to_chars/std::from_chars
- std::atomic<T>::is_always_lock_free
- std::scoped_lock
- std::timespec_get
- 用于 std::chrono::duration 和 std::chrono::time_point 的舍入函数
[编辑] 缺陷报告
[编辑] 编译器支持
[编辑] C++17 核心语言特性
C++17 特性 |
提案 |
GCC |
Clang |
MSVC |
Apple Clang |
EDG eccp |
Intel C++ |
Nvidia HPC C++ (前 PGI)* |
Nvidia nvcc |
Cray |
Embarcadero C++ Builder |
IBM Open XL C++ for AIX |
|
---|---|---|---|---|---|---|---|---|---|---|---|---|---|
DR11:用于直接列表初始化新的 auto 规则 | N3922 | 5 | 3.8 | 19.0 (2015)* | 是 | 4.10.1 | 17.0 | 17.7 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
模板模板参数中的 typename | N4051 | 5 | 3.5 | 19.0 (2015)* | 是 | 4.10.1 | 17.0 | 17.7 | 是* | 11.0 | 10.3 | 17.1.0 | |
移除 三字母词 | N4086 | 5 | 3.5 | 16.0* | 是 | 5.0 | 19.1 | 11.0 | 11.0 | 10.3 | 17.1.0 | ||
嵌套命名空间 定义 | N4230 | 6 | 3.6 | 19.0 (Update 3)* | 是 | 4.12 | 17.0 | 17.7 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
无消息的 static_assert (FTM)* | N3928 | 6 | 2.5 | 19.10* | 是 | 4.12 | 18.0 | 17.7 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
用于命名空间和枚举器的 属性 (FTM)* (FTM)* | N4266 | 4.9 (部分)* 6 |
3.6 | 19.0 (2015)* | 是 | 4.11 | 17.0 | 17.7 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
u8 字符字面量
|
N4267 | 6 | 3.6 | 19.0 (2015)* | 是 | 4.11 | 17.0 | 17.7 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
允许对所有非类型模板参数进行常量求值 (FTM)* | N4268 | 6 | 3.6 | 19.12* | 是 | 5.0 | 19.0.1 | 19.1 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
一元折叠表达式 和空参数包 | P0036R0 | 6 | 3.9 | 19.12* | 是 | 4.14 | 19.0 | 19.1 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
移除 register 关键字的弃用用法 | P0001R1 | 7 | 3.8 | 19.11* | 是 | 4.13 | 18.0 | 17.7 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
移除弃用的 operator++(bool) | P0002R1 | 7 | 3.8 | 19.11* | 是 | 4.13 | 18.0 | 17.7 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
使异常规范成为类型系统的一部分 (FTM)* | P0012R1 | 7 | 4 | 19.12* | 是 | 4.14 | 19.0 | 19.1 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
预处理器条件中的 __has_include | P0061R1 | 5 | 是 | 19.11* | 是 | 4.13 | 18.0 | 17.7 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
DR11:继承构造函数 的新规范 (DR1941 等) (FTM)* | P0136R1 | 7 | 3.9 | 19.14** | 是 | 6.1 | 19.1 | 11.0 | 11.0 | 10.3 | 17.1.0 | ||
具有基类的 聚合类 (FTM)* | P0017R1 | 7 | 3.9 | 19.14* | 是 | 5.0 | 19.0.1 | 19.1 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
折叠表达式 (FTM)* | N4295 | 6 | 3.6 | 19.12* | 是 | 4.14 | 19.0 | 18.1 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
Lambda 捕获 *this (FTM)* | P0018R3 | 7 | 3.9 | 19.11* | 是 | 4.14 | 19.0 | 18.1 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
枚举的直接列表初始化 | P0138R2 | 7 | 3.9 | 19.11* | 是 | 4.14 | 18.0 | 19.1 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
constexpr lambda 表达式 (FTM)* |
P0170R1 | 7 | 5 | 19.11* | 是 | 4.14 | 19.0 | 18.1 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
基于范围的 for 循环 中不同的起始和结束类型 (FTM)* | P0184R0 | 6 | 3.9 | 19.10* | 是 | 4.12 | 18.0 | 17.7 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
[[fallthrough]] 属性 |
P0188R1 | 7 | 3.9 | 19.10* | 是 | 4.13 | 18.0 | 17.7 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
[[nodiscard]] 属性 |
P0189R1 | 7 | 3.9 | 19.11* | 是 | 4.13 | 18.0 | 17.7 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
[[maybe_unused]] 属性 |
P0212R1 | 7 | 3.9 | 19.11* | 是 | 4.13 | 18.0 | 17.7 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
十六进制 浮点字面量 (FTM)* | P0245R1 | 3.0 | 是 | 19.11* | 是 | 4.13 | 18.0 | 17.7 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
不重复使用属性命名空间 | P0028R4 | 7 | 3.9 | 19.11* | 是 | 4.13 | 18.0 | 17.7 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
用于过度对齐数据的 动态内存分配 (FTM)* | P0035R4 | 7 | 4 | 19.12* | 10.0.0* | 4.14 | 19.0 | 19.1 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
具有 auto 类型的非类型模板参数 (FTM)* | P0127R2 | 7 | 4 | 19.14* | 是 | 5.0 | 19.0.1 | 19.1 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
保证的 拷贝省略 (FTM)* | P0135R1 | 7 | 4 | 19.13* | 是 | 5.0 | 19.0.1 | 19.1 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
包含引用成员的类对象的替换 | P0137R1 | 7 | 6 | 19.14* | 是 | 5.0 | 19.1 | 11.0 | 11.0 | 10.3 | 17.1.0 | ||
更严格的 表达式求值顺序 | P0145R3 | 7 | 4 | 19.14* | 是 | 5.0 | 19.0.1 | 19.1 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
结构化绑定 (FTM)* | P0217R3 | 7 | 4 | 19.11* | 是 | 4.14 | 19.0 | 18.1 | 11.0* | 11.0 | 10.3 | 17.1.0 | |
忽略未知 属性 | P0283R2 | 是 | 3.9 | 19.0 (2015)* | 是 | 4.13 | 18.0 | 17.7 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
constexpr if 语句 (FTM)* | P0292R2 | 7 | 3.9 | 19.11* | 是 | 4.14 | 19.0 | 18.1 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
用于 if 和 switch 的初始化语句 | P0305R1 | 7 | 3.9 | 19.11* | 是 | 4.14 | 18.0 | 18.1 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
内联变量 (FTM)* | P0386R2 | 7 | 3.9 | 19.12* | 是 | 4.14 | 19.0 | 18.1 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
移除 动态异常规范 | P0003R5 | 7 | 4 | 19.12* | 是 | 4.14 | 19.0 | 19.1 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
using-声明中的包扩展 (FTM)* | P0195R2 | 7 | 4 | 19.14* | 是 | 5.0 | 19.0 | 19.1 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
DR98:模板模板参数的匹配不包括兼容模板 (FTM)* | P0522R0 | 7 | 4 | 19.12* | 是 | 5.0 | 19.0.1 | 19.1 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
类模板参数推导 (FTM)* | P0091R3 | 7 | 5 | 19.14* | 是 | 5.0 | 19.0.1 | 19.1 | 11.0 | 11.0 | 10.3 | 17.1.0 | |
C++17 特性 |
提案 |
GCC |
Clang |
MSVC |
Apple Clang |
EDG eccp |
Intel C++ |
Nvidia HPC C++ (前 PGI)* |
Nvidia nvcc |
Cray |
Embarcadero C++ Builder |
IBM Open XL C++ for AIX |
[编辑] C++17 库特性
C++17 特性 |
提案 |
GCC libstdc++ |
Clang libc++ |
MSVC STL |
Apple Clang* |
IBM Open XL C/C++ for AIX* |
Intel Parallel STL |
Embarcadero C++ Builder* |
|
---|---|---|---|---|---|---|---|---|---|
std::void_t (FTM)* | N3911 | 6.1 | 3.6 | 19.0 (2015)* | 是 | 17.1.0 | 不适用 | 10.3 | |
std::uncaught_exceptions() (FTM)* | N4259 | 6.1 | 3.7 | 19.0 (2015)* | 是 | 17.1.0 | 不适用 | 10.3 | |
std::size(), std::empty() 和 std::data() (FTM)* | N4280 | 6 | 3.6 | 19.0 (2015)* | 是 | 17.1.0 | 不适用 | 10.3 | |
std::allocator_traits::is_always_equal, noexcept 清理 (FTM)* | N4258 | 6.1 | 3.7 | 19.0 (2015)* | 17.1.0 | 不适用 | |||
std::invoke (FTM)* | N4169 | 6.1 | 3.7 | 19.0 (2015)* | 是 | 17.1.0 | 不适用 | ||
std::map::try_emplace, std::map::insert_or_assign (FTM)* | N4279 | 6.1 | 3.7 | 19.0 (2015)* | 是 | 17.1.0 | 不适用 | ||
std::unordered_map::try_emplace, std::unordered_map::insert_or_assign (FTM)* | N4279 | 6.1 | 3.7 | 19.0 (2015)* | 是 | 17.1.0 | 不适用 | ||
改进 std::pair 和 std::tuple | N4387 | 6.1 | 4 | 19.0 (更新 2)* | 是 | 17.1.0 | 不适用 | 10.3 | |
std::bool_constant (FTM)* | N4389 | 6.1 | 3.7 | 19.0 (2015)* | 是 | 17.1.0 | 不适用 | 10.3 | |
std::shared_mutex (无定时) (FTM)* | N4508 | 6 | 3.7 | 19.0 (更新 2)* | 是 | 17.1.0 | 不适用 | 10.3 | |
std::forward_list, std::list 和 std::vector 的最小不完整类型支持 (FTM)* | N4510 | 3.0 | 3.6 | 18.0* | 是 | 17.1.0 | 不适用 | ||
类型特性 变量模板 (FTM)* | P0006R0 | 7.1 | 3.8 | 19.0 (更新 2)* | 是 | 17.1.0 | 不适用 | 10.3 | |
逻辑运算符类型特性 (FTM)* | P0013R1 | 6.1 | 3.8 | 19.0 (更新 2)* | 是 | 17.1.0 | 不适用 | 10.3 | |
std::as_const (FTM)* | P0007R1 | 7.1 | 3.8 | 19.0 (更新 2)* | 是 | 17.1.0 | 不适用 | ||
用于 std::chrono::duration 和 std::chrono::time_point 的舍入函数 (FTM)* | P0092R1 | 7.1 | 3.8 | 19.0 (更新 2)* | 是 | 17.1.0 | 不适用 | ||
透明 std::owner_less (std::owner_less<void>) (FTM)* | P0074R0 | 7.1 | 3.8 | 19.0 (更新 2)* | 是 | 17.1.0 | 不适用 | ||
std::not_fn (FTM)* | P0005R4 P0358R1 |
7.1 | 3.9 | 19.12* | 是 | 17.1.0 | 不适用 | 10.3 | |
并行算法和 执行策略 (FTM)* (FTM)* | P0024R2 | 9* | 17 (部分)* |
19.14* | 18.0* | ||||
std::clamp() (FTM)* | P0025R1 | 7 | 3.9 | 19.0 (Update 3)* | 10.0.0* | 17.1.0 | 不适用 | 10.3 | |
(无异常)可交换特性 (FTM)* | P0185R1 | 7.1* | 3.9 | 19.0 (Update 3)* | 10.0.0* | 17.1.0 | 不适用 | 10.3 | |
多态内存资源 (FTM)* | P0220R1 | 9.1 | 16 | 19.13* | 15.0.0* | 17.1.1 | 不适用 | 10.3 | |
std::apply (FTM)* | P0220R1 | 7.1 | 16 | 19.13* | 是 | 17.1.0 | 不适用 | ||
搜索器 (FTM)* | P0220R1 | 7.1 | 16 | 19.13* | 是 | 17.1.0 | 不适用 | ||
std::sample (FTM)* | P0220R1 | 7.1 | 16 | 19.13* | 是 | 17.1.0 | 不适用 | ||
数学特殊函数 (FTM)* | P0226R1 | 7 | 19.14* | 17.1.1 | 不适用 | 10.3 | |||
constexpr std::addressof (FTM)* | LWG2296 | 7.1 | 是 | 19.0 (Update 3)* | 17.1.0 | 不适用 | |||
constexpr 用于 std::reverse_iterator, std::move_iterator, std::array 和范围访问 (FTM)* | P0031R0 | 7.1 | 4 | 19.11* | 是 | 17.1.0 | 不适用 | ||
constexpr std::atomic<T>::is_always_lock_free (FTM)* | P0152R1 | 7.1 | 3.9 | 19.11* | 是 | 17.1.0 | 不适用 | ||
std::enable_shared_from_this::weak_from_this (FTM)* | P0033R1 | 7.1 | 3.9 | 19.12* | 是 | 17.1.0 | 不适用 | ||
std::hypot 的 3 参数重载 (FTM)* | P0030R1 | 7.1 | 3.9 | 19.14* | 是 | 17.1.0 | 不适用 | ||
std::byte (FTM)* | P0298R3 | 7 | 5 | 19.11* | 是 | 17.1.1 | 不适用 | 10.3 | |
std::string_view (FTM)* | N3921 P0220R1 P0254R2 P0403R1 |
7.1 | 4 | 19.10* (partial)* 19.11** |
10.0.0* | 17.1.0 | 不适用 | 10.3 | |
std::any (FTM)* | P0220R1 P0032R3 |
7.1 | 4 | 19.10* | 10.0.0* | 17.1.0 | 不适用 | 10.3 | |
std::optional (FTM)* | P0220R1 | 7.1 | 4 | 19.10* | 10.0.0* | 17.1.0 | 不适用 | 10.3 | |
C11 标准库的主要部分 | P0063R3 | 9.1 | 7 | 19.0 (2015)* (部分)* |
10.0.0* | 17.1.1 | 不适用 | ||
拼接 Maps 和 Sets (FTM)* | P0083R3 | 7 | 8 | 19.12* | 10.0.0* | 17.1.1 | 不适用 | ||
一些 容器 的 emplace* 函数的返回类型从 void 更改为引用 | P0084R2 | 7.1 | 4.0 | 19.11* | 是 | 17.1.0 | 不适用 | ||
std::variant (FTM)* | P0088R3 | 7.1 | 4 | 19.10* | 10.0.0* | 17.1.0 | 不适用 | 10.3 | |
std::make_from_tuple() (FTM)* | P0209R2 | 7.1 | 3.9 | 19.10* | 是 | 17.1.0 | 不适用 | 10.3 | |
std::has_unique_object_representations (FTM)* | P0258R2 | 7.1 | 6 | 19.11* | 是 | 17.1.1 | 不适用 | 10.3 | |
std::gcd() 和 std::lcm() (FTM)* | P0295R0 | 7 | 4 | 19.11* | 是 | 17.1.0 | 不适用 | 10.3 | |
CWG issue 1776:替换包含引用成员的类对象 (std::launder) (FTM)* | P0137R1 | 7.1 | 6 | 19.14* | 是 | 17.1.0 | 不适用 | ||
扩展内存管理工具 (FTM)* | P0040R3 | 7.1 | 4 | 19.11* | 17.1.0 | 不适用 | |||
shared_ptr::weak_type (FTM)* | P0163R0 | 7.1 | 3.9 | 19.10* | 是 | 17.1.0 | 不适用 | ||
基本字符串转换:std::to_chars / std::from_chars (FTM)* | P0067R5 | 8* 11 |
7* 14* 20* |
19.14** 19.24* |
10.0.0** | 17.1.1* | 不适用 | 10.3* | |
支持数组的 std::shared_ptr 和 std::weak_ptr | P0414R2 | 7 | 11 | 19.12* | 12.0.0* | 17.1.1 | 不适用 | 10.3 | |
std::chrono::duration 和 std::chrono::time_point 所有成员函数的 Constexpr (FTM)* | P0505R0 | 7.1 | 4 | 19.11* | 是 | 17.1.1 | 不适用 | ||
std::shared_ptr<T[]> (FTM)* | P0497R0 | 7.1 | 11 | 19.12* | 17.1.1 | 不适用 | |||
constexpr std::char_traits (FTM)* | P0426R1 | 8.1 | 4 | 19.14* | 是 | 17.1.1 | 不适用 | ||
文件系统库 (std::filesystem) (FTM)* | P0218R1 P0219R1 |
8 | 7 | 19.14* | 11.0.0* | 17.1.1 | 不适用 | 10.3 | |
硬件干扰大小 (FTM)* | P0154R1 | 12.1 | 15 (部分)* 19 |
19.11* | 不适用 | 10.3 | |||
std::scoped_lock (FTM)* | P0156R2 | 7 | 5 | 19.11* | 是 | 17.1.1 | 不适用 | 10.3 | |
std::is_aggregate (FTM)* | LWG2911 | 7 | 5 | 19.15* | 是 | 17.1.1 | 不适用 | 10.3 | |
std::is_invocable, std::invoke_result (FTM)* | P0604R0 | 7.1 | 是 | 19.11* | 是 | 17.1.1 | 不适用 | ||
DR17: std::hash<std::filesystem::path> | LWG3657 | 11.4 | 17 | 19.32* | 不适用 | ||||
C++17 特性 |
提案 |
GCC libstdc++ |
Clang libc++ |
MSVC STL |
Apple Clang* |
IBM Open XL C/C++ for AIX* |
Intel Parallel STL |
Embarcadero C++ Builder* |
[编辑] 备注
- 截至 2020-11-20,Oracle Developer Studio 的最新版本是 12.6。其文档未提及 C++17。
- Cray 编译器可能在 11.0 之前就支持某些特性。该版本是它成为 Clang 的派生版本,获得了基础编译器所有相关语言特性支持的时候。请参阅 Cray/HPE 文档 S-2179。
* - 将鼠标悬停在标有星号 * 的单元格上可查看额外的弹出式注释。
DRnn - “DR”后的数字 nn 表示该缺陷报告(Defect Report)所适用的 C++ 修订版本,例如 DR20 → C++20。
[编辑] 外部链接
1. | C++17 - 维基百科 |
2. | C++17 示例 |