C++11/14新特性学习笔记
闲暇之余,在B站看到了侯捷老师的C++2.0课程视频,通过学习,了解了C++2.0的一些新特性。侯捷老师的课程从语言层面和标准库层面分别介绍了这些新特性。
语言层面
通过输出打印宏定义“__cplusplus”的值可以查看编译器提供的版本。若数字大于等于201103,则表示支持C++11。
#include <iostream>
using std::cout;
using std::endl;
int main(int argc, char* argv[]){
cout << __cplusplus << endl;
return 0;
}
//--std=c++2a -> 201709
//--std=c++17 -> 201703
//--std=c++14 -> 201402
//--std=c++11 -> 201103
//我的环境下不加编译选项 -> 201402
关于模板的改进
模板的右尖括号
C++1.0是不允许两个右尖括号连在一起出现的,会被认为是右移操作符,所以在C++11之前的模板会用空格将两个右尖括号分开。
C++11之后就不需要这样了。
#include <iostream> #include <vector> using std::cout; using std::endl; using std::vector; int main(int argc, char* argv[]){ cout << __cplusplus << endl; vector<vector<char>> a; return 0; } //--std=c++98 //error: '>>' should be '> >' within a nested template argument list
别名模板
template <模板形参类表> using 标识符 attr(可选) = 类型标识; ```
C++11引入了using定义别名的方法,可替代typedef,使得定义别名更加通俗易懂。typedef是不接受参数的,也就无法使用别名模板的模板实参替换类型标识。
与任何模板声明相似,别名模版仅可声明于类作用域或命名空间作用域。
在推导模板模板形参时,模板实参推导始终不推导别名模板。不可能部分特化或显式特化别名模板。
```
template <class T>
struct Alloc {};
template <class T>
using Vec = std::vector<T, Alloc<T>>;
Vec<int> coll; //等同于std::vector<int, Alloc<int>>
关键词 nullptr 代表指针字面量。它是 std::nullptr_t 类型的纯右值。存在从 nullptr 到任何指针类型及任何成员指针类型的隐式转换。 std::nullptr_t 是空指针字面量 nullptr 的类型。它是既非指针类型亦非指向成员指针类型的独立类型。 宏 对于变量,指定要从其初始化器自动推导出其类型。 对于函数,指定要从其 return 语句推导出其返回类型。 (C++14 起) 对于非类型模板形参,指定要从实参推导出其类型。 (C++17 起)
template 指针字面量:nullptr
#include <cstddef>
#include <iostream>
template<class F, class A>
void Fwd(F f, A a)
{
f(a);
}
void g(int* i)
{
std::cout << "Function g called\n";
}
int main()
{
g(NULL); // 良好
g(0); // 良好
Fwd(g, nullptr); // 良好
// Fwd(g, NULL); // 错误:不存在函数 g(int)
}
NULL
是实现定义的空指针常量,C++11后可能为零值整数字面量,或为 std::nullptr_t 类型纯右值。#define NULL 0
//C++11起
#define NULL nullptr
占位符类型说明符:auto
//auto 常用于无名类型,例如 lambda 表达式的类型
auto lambda = [](int x) { return x + 3; };
//auto 常用于迭代器类型
vecotr<int> a;
auto i = a.begin();
template<auto n> // C++17 auto 形参声明
auto f() -> std::pair<decltype(n), decltype(n)> // auto 不能从花括号初始化器列表推导
{
return {n, n};
}