auto
用于初始化列表
1 |
auto m = {10} |
上述代码,在C++17
之前,对于m的类型的推断,编译器给出的结果将不会是我们的预期的。
它会推断为std::initializer_list<int>
,而不是int
。
同样,下面的代码也是会被推断为std::initializer_list<int>
的:
1 |
auto list = {1,2,3,4}; |
在C++17
新标准中,这种类型推断的规则发生了变化。
1 2 3 4 5 6 7 8 9 10 11 |
//C++11/14 auto i {10};//initializer_list<int> auto pi = {3.1415};//initializer_list<double> auto list1{1,2,3};//initializer_list<int> auto list2 = {4,5,6};//initializer_list<int> //C++17 auto i {10};//int auto pi = {3.1415};//initializer_list<double> auto list1{1,2,3};//error:does not compile auto list2 = {4,5,6};//initializer_list<int> |
结构化绑定
- 解构元组、结构体或数组为独立变量
1 2 3 4 5 6 7 |
#include <tuple> std::tuple<int, double> getData() { return {42, 3.14}; } int main() { auto [num, val] = getData(); // 结构化绑定 // num = 42, val = 3.14 } |
内联变量
- 允许头文件中直接定义全局变量
1 2 |
// header.h inline int globalVar = 42; // 允许多次包含,链接时合并 |
折叠表达式
- 简化变参模板的参数包展开
1 2 3 4 5 6 7 8 |
template<typename... Args> auto sum(Args... args) { return (... + args); // 展开为 (arg1 + (arg2 + ...)) } int main() { sum(1, 2, 3); // 返回 6 } |
类模板参数推导
- 自动推导模板参数类型
1 2 |
std::pair p(42, "hello"); // 推导为 std::pair<int, const char*> std::vector vec{1, 2, 3}; // 推导为 std::vector<int> |
constexpr if
- 编译期条件判断,简化模板代码
1 2 3 4 5 6 7 8 |
template<typename T> auto process(T val) { if constexpr (std::is_integral_v<T>) { return val * 2; } else { return val.substr(1); } } |
贯穿[[fallthrough
]]
switch
语句中,如果在一个case
后面漏掉了break
,那么程序还会继续执行到下一个case
的内容。-
在
C++17
新标准中,如果我们需要在某个地方故意使用贯穿行为,我们可以在该case
原本break
的位置写上[[fallthrough]]
。来告诉编译器和阅读代码的人,我们在这个地方故意使用了贯穿行为。1234567891011switch(ticket_number){case 0:cout << "aet!!!" << endl;[[fallthrough]];case 1:cout << "lif_wang!!!" << endl;break;default:break;}
[[nodiscard]]
- 强调返回值必须处理
1 2 |
[[nodiscard]] int compute(); compute(); // 编译警告(若忽略返回值) |
[[maybe_unused]]
- 抑制未使用变量警告
1 |
void func([[maybe_unused]] int x) {} |
控制在if()&&switch()
中初始化变量并使用
-
因为认为将变量的作用域控制到使用它们的区域,是一种良好的编程风格。
-
所以,下方代码中,
lower
的作用域就有点超出它的范围了(因为它仅仅是被用来在if()
里面判断一下的,其他地方根本就不会再使用到它)。123auto lower {static_cast<char>(std::tolower(input))};if(lower >= 'a' && lower <= 'z')cout << "you've entered the letter" << endl; -
所以,为了改进这一点,在
C++17
新标准中,专门引入了新的语法。1if(initialization;condition) ... -
上述代码可修改为:
12if(auto lower {static_cast<char>(std::tolower(input))};lower >= 'a' && lower <= 'z')cout << "you've entered the letter" << endl; -
而且,为了完整起见,在
C++17
新标准,还为switch语句添加了类似的语法。1switch(initialization;condition) {...}
1 2 3 4 5 |
if (int x = computeValue(); x > 10) { // 使用 x } else { // x 仍在此作用域 } |
std::size()
- 在必要时,我们需要确定数组的大小。
-
在C++17新标准,在标准库array头文件中提供了
size()
函数。size()
函数不只适用于数组,还可以用来获取标准库定义的任何元素集合的大小(包括vector
,array
)- 编译器会通过数组定义中初始值的数量来计算数组的大小。
123456789#include <iostream>#include <array>//for std::size()using namespace std;int main(){int values[] {2,3,4,5,6,7,8,9};cout << size(values) << endl;return 0;}
关于数组的维数
- 在
C++17
中,不允许在运行期间指定数组的维数,数组的维数必须是一个编译器能计算的常量表达式。- 但是,在目前的一些
C++
编译器中,是允许在运行期间指定数组的维数的。因为目前的C
标准C11
允许这么做,而C++
编译器一般也编译C
代码。
- 但是,在目前的一些
string
的c_str()
和data()
概述
1 2 |
string ptr_string {"many money"}; const char * ptr_c_str = ptr_string.c_str(); |
-
类似上面,可以将
string
转换为const char *
的C
字符串。 -
第二种方法就是,从
C++17
标准开始,我们使用string
对象的data()
函数,会得到一个非const char *
的指针。- 在
C++17
之前,使用data()
得到的是一个const char *
指针
1char * ptr_data = ptr_string.data();区别
- 在
-
C++11
之前c_str
返回一个以'\0'
结尾的C
风格的字符串;data
返回一个const char*
指针。
-
C++11
之后- 两者都返回
const char*
指针。
- 两者都返回
-
C++17
之后c_str
返回const char*
指针;data
返回char*
指针。
关于Unicode
- 在
C++17
标准中,弃用了标准库提供的大部分在各种Unicode
编码之间进行转换的功能。- 推荐使用:
ICU
库 - 或者在
ICU
基础之上的Boost.Locale
库
- 推荐使用:
string_view
字符串视图:新的const string
引用
概述
- 在
C++17
标准库新增的string_view
头文件中,定义了一个string_view
的类型。- 此类型的值的行为非常类似
const std::string
类型的值。
- 此类型的值的行为非常类似
-
但有一个重要区别就是:
- 任何时候都不能通过它们的公共接口修改它们封装的字符串。
- 也就是说,某种程度上,
string_view
具有固定的常量特性
我们可以查看view
但不能修改string_view
的字符。 - 这种局限性就意味着,这些对象和
std::string
是不同的,我们不需要自己操作字符数组的副本。
相反,它只需要指向某个实际的string
对象、某个字符串字面量或者其他任何字符数组中存储的任何字符序列即可。
因为它不涉及复制字符数组,所以它的开销是很低的。
12345678910111213void find_words(vector<string> & words,string_view str,string_view separators){size_t start{str.find_first_not_of(separators)};size_t end{};while(start != string::npos){end = str.find_first_of(separators,start+1);if(end == string::npos)end = str.length();words.push_back(str.substr(start,end-start));start = str.find_first_not_of(separators,end+1);}}
1 2 3 4 5 |
#include <string_view> void print(std::string_view sv) { // 高效读取,不复制数据 } print("Hello World"); // 隐式转换 |
- 特点:
tring_view
没有c_str()
函数来转换为一个const char *
数组,但它提供了data()
函数- 不能使用加法运算符来连接
string_view
,要这么操作的话,必须先将string_view
转换成string
理解
- 只读试图
string_view
确实不可修改底层字符串内容(类似const std::string&
),所有修改操作(如remove_prefix()
、remove_suffix()
)仅调整视图范围,不改变原始数据
- 零拷贝开销
- 它仅存储指向原始字符串的指针和长度,构造时不复制数据,适合高频操作大字符串的场景
std::optional
概述
- 在
C++17
标准中,标准库提供了std::optional<>
,它可以代替下述情况的可选值的隐式编码。通过使用这种辅助类型,可以使用optional<int>
显示声明任何可选的int
值。 - 当调用函数想让被调用函数使用其默认设置,或者因为无法计算实际的值而返回时,传统方法是使用某个或某些特定的值
- 就比如在
string
里面,如果在一个串里查找,find
函数如果没找到,返回string::npos
- 就比如在
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 |
#include <iostream> #include <optional> #include <string_view> using std::optional; using std::string_view; optional<size_t> find_last(string_view string,char to_find,optional<size_t> start_index = std::nullopt) { if(string.empty()) return std::nullopt; size_t index = start_index.value_or(string.size()-1); while(true) { if(string[index] == to_find) return index; if(index == 0) return std::nullopt; --index; } } int main() { const auto string = "aetlifwang"; const optional<size_t> found_a{find_last(string,'a')}; if(found_a) cout << "found the last a at index" << *found_a << endl; const auto found_b{find_last(string,'b')}; if(found_b.has_value()) cout << "found the last b at index" << found_b.value() << endl; //const size_t found_c{find_last(string,'c')};//error,不能转换为size_t const auto found_early_i {found_last(string,'i',10)}; if(found_early_i != std::nullopt) cout << "found an early i at index" << *found_early_i << endl; } |
- 关于
nullopt
,它是标准库定义的一个特殊常量,用于初始化好没有为T
赋值的optional<T>.
总结理解
- 表示可能存在或不存在的值
1 2 3 4 5 6 7 8 9 |
#include <optional> std::optional<int> find(int key) { if (key == 42) return 100; return std::nullopt; // 无值 } if (auto val = find(42)) { // 使用 *val } |
检查头文件是否可用
- 标准库的每个新版本都提供了许多新的头文件,在这些头文件中定义了新的功能。
-
有的时候,我们需要使用多个编译器编译运行,可能是相同编译器的多个版本,或者针对不同目标平台的不同编译器。
- 这时候,需要有一种方式,在编译时检查当前编译器支持哪些功能,以便相应的启动或禁用代码的不同版本。
C++17
新标准,引入了__has_include()
宏,可用来检查任何头文件是否可用。
123456789#if __has_include(<SomeStandardLibaryHeader>)#include <SomeStandardLibaryHeader>//...#elif __has_include("SomeHeader.h")#include "SomeHeader.h"//...#else#error("need at least SomeStandardLibaryHeader or SomeHeader.h")#endif
本文为原创文章,版权归Aet所有,欢迎分享本文,转载请保留出处!
你可能也喜欢
- ♥ C++程序高级调试与优化_第一篇07/20
- ♥ C++_关于对象的具体初始化顺序11/30
- ♥ C++标准模板库编程实战_算法和随机数12/08
- ♥ C++并发编程_概念了解05/07
- ♥ C++并发编程_同步并发(Condition_variable)05/21
- ♥ 深入理解C++11:C++11新特性解析与应用 三01/05