当前位置: 首页 > news >正文

[C++] C++11新增

一、列表初始化

C++98:

在C++98中,标准允许使用花括号{}对数组元素进行统一的列表初始值设定。

struct Simple1
{int _a;int _b;
};//C++98
int main()
{int a1[] = { 1,2,3,4,5,6 };int a2[7] = { 0 };//本质是类型转换(构造+拷贝构造 -> 优化 直接构造)Simple1 s1 = { 1,2 };return 0;
}

C++11:

C++11扩大了用大括号括起的列表(初始化列表)的使用范围,使其可用于所有的内置类型用户自定义的类型,使用初始化列表时,可添加等号(=),也可不添加。(列表初始化可以在{}之前使用等号,其效果与不使用=没有什么区别。)

实际是因为C++11新增了一个initializer_list,initializer_list是系统自定义的类模板,该类模板中主要有三个方法:begin()、end()迭代器以及获取区间中元素个数的方法size()。

多个对象想要支持列表初始化,只需给该类(模板类)添加一个带有initializer_list类型参数的构造函数即可。


//C++11
//一切都可以用列表初始化
int main()
{// 内置类型变量int x1 = { 1 };int x2{ 1 };int x3 = 1 + 1;int x4 = { 1 + 1 };int x5{ 1 + 1 };// 数组int arr1[5]{ 1,2,3,4,5 };int arr2[]{ 1,2,3,4,5 };// 动态数组,在C++98中不支持int* arr3 = new int[5]{ 1,2,3,4,5 };// 标准容器vector<int> v1 = { 1,2,3,4,5 };vector<int> v{ 1,2,3,4,5 };map<int, int> m1 = { {1,1}, {2,2,},{3,3},{4,4} };map<int, int> m{ {1,1}, {2,2,},{3,3},{4,4} };//标准库支持单个对象的列表初始化Simple1 s{ 1, 2 };return 0;
}

二、声明

1、auto

C++11中,可以使用auto来根据变量初始化表达式类型推导变量的实际类型,可以给程序的书写提供许多方便。将程序中it的类型换成auto,程序可以通过编译,而且更加简洁。

int main()
{map<string, string> m{ {"menu", "菜单"}, {"delete","删除"} };// 使用迭代器遍历容器, 迭代器类型太繁琐//map<string, string>::iterator it = m.begin();//使用autoauto it = m.begin();while (it != m.end()){cout << it->first << " " << it->second << endl;++it;}return 0;
}

2、decltype

auto使用的前提是:必须要对auto声明的类型进行初始化,否则编译器无法推导出auto的实际类型。但有时候可能需要根据表达式运行完成之后结果的类型进行推导,因为编译期间,代码不会运行,此时auto也就无能为力。

而decltype是根据表达式的实际类型推导出定义变量时所用的类型。这个类型可以用来做实例化模版实参或者再定义对象。

int main()
{int a = 11;float b = 10.11;auto ret = a * b;vector<decltype(ret)> v{ 12,12.12,14.1 };for (auto e : v){cout << e << endl;}return 0;
}

三、范围for

范围for在底层实际是被替换成了迭代器。

int main()
{vector<int> v{ 1,4,6,2,5,11 };for (auto e : v){cout << e << " ";}cout << endl;return 0;
}

 

四、智能指针

由于智能指针内容较多,请查看智能指针

五、新增容器

静态数组array

array<int,10> arr = { 1,2,3,4,5 };

forward_list

forward_list实际上就是单链表,区别于双向链表list

unordered系列

(unordered_set/unordered_map)

参考:哈希

六、右值引用

1、左值与右值:

一般认为:

  •  普通类型的变量,因为有名字,可以取地址,都认为是左值。
  • const修饰的常量,不可修改,只读类型的,理论应该按照右值对待,但因为其可以取地址(如果只是const类型常量的定义,编译器不给其开辟空间,如果对该常量取地址时,编译器才为其开辟空间),C++11认为其是左值。
  • 如果表达式的运行结果是一个临时变量或者对象,认为是右值。
  • 如果表达式运行结果或单个变量是一个引用则认为是左值。

C++11对右值进行了严格的区分:

  • C语言中的纯右值,比如:a+b, 100
  • 将亡值。比如:表达式的中间结果、函数按照值的方式进行返回。

2、左值引用与右值引用

左值引用:

int main()
{int* p1 = new int[10]{ 1,2,3,4,5,6,7,8,9,99 };int*& r1 = p1;int a = 1;int& r2 = a;const int& r3 = 10;return 0;
}

右值引用:

int main()
{int&& r1 = 10;int&& r2 = 1 + 3;int i = 11;int&& r3 = i + 11;int&& rr = move(i);return 0;
}

注:右值引用可以给move(左值)取别名。不过,不要轻易使用move,有时候会产生意想不到的结果。

3、移动构造/移动赋值

C++11中新增了两个默认成员函数:移动构造函数移动赋值运算符重载。

如果我们没有实现移动构造函数,且没有实现析构函数、拷贝构造、拷贝赋值重载中的任意一个(都没有写),那么编译器会 自动生成一个默认移动构造。默认生成的移动构造函数,对于内置类型成员,会逐成员按字节拷贝,自定义类型成员,则看它是否实现了移动构造,若实现了则调用,否则调用拷贝构造。(移动赋值重载与其完全类似

不过,在C++11中,可以在默认函数定义或者声明时加上=default,从而显式的指示编译器生成该函数的默认版本,用=default修饰的函数称为显式缺省函数。(强制编译器生成)

此外,如果能想要限制某些默认函数的生成,在C++98中,是该函数设置成private,并且不给定义,这样只要其他人想要调用就会报错。在C++11中更简单,只需在该函数声明加上=delete即可,该语法指示编译器不生成对应函数的默认版本,称=delete修饰的函数为删除函数。

C++11提出了移动语义概念,即:将一个对象中资源移动到另一个对象中。

这样就解决了 由于函数返回值的生命周期(函数返回时,一般是创建一个临时对象,用该临时对象构造接收函数返回值的变量)问题,无法使用引用返回 的问题。

而在C++11中如果需要实现移动语义,必须使用右值引用。

如:

String(String&& s)
: _str(s._str)
{s._str = nullptr;
}

注:

  • 移动构造函数的参数不能设置成const类型的右值引用,因为资源无法转移而导致移动语义失效。
  • 在C++11中,编译器会为类默认生成一个移动构造,该移动构造为浅拷贝,因此当类中涉及到资源管理时,用户必须显式定义自己的移动构造。
  • 右值被右值引用引用后的属性是左值。因为右值不能直接修改,但是右值被右值引用后,需要被修改,否则无法实现移动构造和移动赋值。

当需要用右值引用引用一个左值时,可以通过move函数将左值转化为右值。C++11中std::move()函数位于 头文件中,该函数名字具有迷惑性,它并不搬移任何东西,唯一的功能就是将一个左值强制转化为右值引用,然后实现移动语义。

4、完美转发

完美转发:

在函数模板中,完全依照模板的参数的类型,将参数传递给函数模板中调用的另外一个函数:

函数模板在向其他函数传递自身形参时,如果相应实参是左值,它就应该被转发为左值;如果相
应实参是右值,它就应该被转发为右值。

C++11通过forward函数来实现完美转发:

void Fun(int& x) 
{cout << "lvalue ref" << endl;
}void Fun(int&& x) 
{ cout << "rvalue ref" << endl;
}void Fun(const int& x)
{cout << "const lvalue ref" << endl; 
}void Fun(const int&& x)
{ cout << "const rvalue ref" << endl; 
}template<typename T>
void PerfectForward(T&& t)
{ Fun(forward<T>(t)); 
}
int main()
{PerfectForward(10); // rvalue refint a;PerfectForward(a); // lvalue refPerfectForward(move(a)); // rvalue refconst int b = 8;PerfectForward(b); // const lvalue refPerfectForward(move(b)); // const rvalue refreturn 0;
}

 七、可变参数模版

比如list中的emplace_back就是使用了它。

 输出大小:

//Args是一个模版参数包,args是一个函数形参参数包
//声明一个模版参数包Args...args,这个参数包中可以包含0到任意个模版参数
template<class ...Args>
void show_list(Args... args)
{cout << sizeof...(args) << endl;
}int main()
{show_list(1, 1, 1);show_list('a',"aa");show_list(1.1, 'a');show_list(1, 1.1, 'a', "xx");return 0;
}

输出每个元素: 

void _show_list()
{cout << endl;
}
//编译时推演
//第一个模版参数依次解析获取参数值
template<class T,class ...Args>
void _show_list(const T& val, Args ...args)
{cout << val << " ";_show_list(args...);
}template<class ...Args>
void show_list(Args... args)
{_show_list(args...);
}int main()
{show_list(1, 1, 1);show_list('a',"aa");show_list(1.1, 'a');show_list(1, 1.1, 'a', "xx");return 0;
}

八、 lambda表达式

lambda表达式解决了需要写仿函数重载operator()的问题,尤其每次比较的逻辑不一样,需要去实现多个类,特别是相同类的命名。

语法:

lambda表达式书写格式:

[capture-list] (parameters) mutable -> return-type { statement }
 lambda表达式各部分说明:

  • [capture-list] : 捕捉列表,该列表总是出现在lambda函数的开始位置,编译器根据[]来判断接下来的代码是否为lambda函数,捕捉列表能够捕捉上下文中的变量供lambda函数使用。
  • (parameters):参数列表。与普通函数的参数列表一致,如果不需要参数传递,则可以连同()一起省略
  • mutable:默认情况下,lambda函数总是一个const函数,mutable可以取消其常量性。使用该修饰符时,参数列表不可省略(即使参数为空)。
  • ->returntype:返回值类型。用追踪返回类型形式声明函数的返回值类型,没有返回值时此部分可省略。返回值类型明确情况下,也可省略,由编译器对返回类型进行推导。
  • {statement}:函数体。在该函数体内,除了可以使用其参数外,还可以使用所有捕获到的变量。

注意:

  • 在lambda函数定义中,参数列表和返回值类型都是可选部分,而捕捉列表和函数体可以为空。因此C++11中最简单的lambda函数为:[]{}; 该lambda函数不能做任何事情。
  • 捕捉列表描述了上下文中哪些数据可以被lambda使用,以及使用的方式是传值还是传引用。
    [var]:表示值传递方式捕捉变量var
    [=]:表示值传递方式捕获所有父作用域中的变量(包括this)
    [&var]:表示引用传递捕捉变量var
    [&]:表示引用传递捕捉所有父作用域中的变量(包括this)
    [this]:表示值传递方式捕捉当前的this指针
  • 父作用域指包含lambda函数的语句块
  • 语法上捕捉列表可由多个捕捉项组成,并以逗号分割。
  • 捕捉列表不允许变量重复传递,否则就会导致编译错误。 
  • 在块作用域以外的lambda函数捕捉列表必须为空。
  • 在块作用域中的lambda函数仅能捕捉父作用域中局部变量,捕捉任何非此作用域或者非局部变量都会导致编译报错。
  • lambda表达式之间不能相互赋值,即使看起来类型相同

使用:

int main()
{int a = 1;int b = 2;cout << a << " " << b << endl;auto f1 = [](int& a, int& b){int tmp = a;a = b;b = tmp;};f1(a, b);cout << a << " " << b << endl;return 0;
}

int main()
{int a = 1;int b = 2;cout << a << " " << b << endl;//通过捕捉列表,传引用auto f2 = [&a, &b] {int tmp = a;a = b;b = tmp;};f2();cout << a << " " << b << endl;return 0;
}
class Test
{
public:void func(){auto f = [=]{cout << _a << " " << _b << endl;};}private:int _a = 2;int _b = 4;
};int main()
{Test t;t.func();return 0;
}

九、包装器(适配器)

1、function

function包装器包装的是:函数指针(类型用起来反人类)、仿函数(需在全局定义)、lambda(类型对我们是匿名的)中的任意一个。

bool Comp(int& a, int& b)
{return a < b;
}struct Comps
{bool operator()(int& a, int& b){return a < b;}
};#include<functional>
#include<map>
int main()
{auto complambda = [](int& a, int& b) ->bool{return a < b;};map < string, function<bool(int&, int&)>> m{{"函数指针",Comp},{"仿函数",Comps()},{"lambda",complambda}};int x = 1;int y = 2;cout << m["函数指针"](x, y) << endl;cout << m["仿函数"](x, y) << endl;cout << m["lambda"](x, y) << endl;return 0;
}

包装成员函数:

静态成员函数(static)可以不用加“&”。

成员函数取地址比较特殊,需要加上类域和&。

struct AAA
{
public:static void A(int a){a = 0;}void AA(float a){a = 1.1;}};int main()
{function<void(int)> f1 = AAA::A;f1(1);function<void(AAA*, float)> f2 = &AAA::AA;AAA a;f2(&a, 2.2); //注意不能使用匿名对象,因为右值不可以取地址function<void(AAA, float)> f3 = &AAA::AA;f3(AAA(), 2.2);return 0;
}

2.bind

bind是一个函数模板,它就像一个函数包装器(适配器),接受一个可调用对象(callable object),生成一个新的可调用对象来“适应”原对象的参数列表。

struct AAA
{
public:static void A(int a){cout << a << endl;}void AA(float a){cout << a << endl;}};void aaa(int a, int b)
{cout << a << " " << b << endl;
}int main()
{function<void(int)> f1 = AAA::A;f1(1);function<void(AAA*, float)> f2 = &AAA::AA;AAA a;f2(&a, 2.2); //注意不能使用匿名对象,因为右值不可以取地址function<void(AAA, float)> f3 = &AAA::AA;f3(AAA(), 2.2);//调整传参顺序function<void(int, int)> ff1 = bind(aaa, placeholders::_2, placeholders::_1);ff1(3, 5);//调整参数个数function<void(int)> ff2 = bind(aaa, 22, placeholders::_1);ff2(5);function<void(float)> ff3 = bind(&AAA::AA, AAA(), placeholders::_1);ff3(100.1);return 0;
}

可以将bind函数看作是一个通用的函数适配器,它接受一个可调用对象,生成一个新的可调用对
象来“适应”原对象的参数列表。
调用bind的一般形式:auto newCallable = bind(callable,arg_list);
其中,newCallable本身是一个可调用对象,arg_list是一个逗号分隔的参数列表,对应给定的
callable的参数。当我们调用newCallable时,newCallable会调用callable,并传给它arg_list中
的参数。
arg_list中的参数可能包含形如_n的名字,其中n是一个整数,这些参数是“占位符”,表示
newCallable的参数,它们占据了传递给newCallable的参数的“位置”。数值n表示生成的可调用对
象中参数的位置:_1为newCallable的第一个参数,_2为第二个参数,以此类推。

http://www.lryc.cn/news/409628.html

相关文章:

  • 802.11 wireshark 抓包
  • vscode 调试web后端
  • JAVA默写单词小程序
  • 认知、情绪、情感、意志、人格
  • 解析capl文件生成XML Test Module对应的xml工具
  • Java中的反射是怎么回事?
  • 07 STM32寄存器开发基础-中断编程
  • 聚簇和非聚簇索引/Btree和B+tree
  • 清华学姐熬夜肝了15天的软件测试面试题出炉(附答案)建议收藏!
  • Docker 安装指南
  • 系统架构设计师 - 知识产权与标准化
  • 【Python】Facebook开源时间序列数据预测模型Prophet
  • Spring 常用的三种拦截器详解
  • 微前端概念
  • FFmpeg实战 - 解复用解码
  • Jmeter混合压测(2407)
  • Prometheus各类监控及监控指标和告警规则
  • G120 EPos配置方案及应用场景
  • 定制化爬虫管理:为企业量身打造的数据抓取方案
  • Javascript面试基础6【每日更新10】
  • CTF Web信息搜集 25000字详解
  • MSPM0G3507之电赛小车
  • linux运维一天一个shell命令之vmstat详解
  • 前端开发调试工具推荐分类整理
  • http协议与nginx
  • 一款国外开发的高质量WordPress下载站模板主题
  • Laravel为什么会成为最优雅的PHP框架
  • 孤儿进程的例子
  • CSS前端面试题——怎么用CSS实现一个宽高自适应的正方形?
  • 谷粒商城实战笔记-56~57-商品服务-API-三级分类-修改-拖拽功能完成