从零开始的C++(二十一)

C++11

1.列表初始化:

//允许以下代码正确运行

int a[]{1,2,3};//效果与int a[]={1,2,3}一致

即允许省略等于号。同时,允许用花括号对所有自定义类型和内置类型进行初始化,而非以前花括号只能对数组进行初始化。利用花括号对自定义类型初始化时,相当于调用了其构造函数。

2.std::initializer_list

一般用作构造函数的参数,接受一系列的初始值,并以此赋给每个对象。

因此,允许以下操作:


map<int,string>v={{1,"one"},{2,"two"}};
//其中{1,"one"}和{2,"two"}均对应一个pair<int,string>的对象,用于给map赋值。

3.decltype

可以获得变量的类型,并且可以用这个类型构造变量(与typeid不同,typeid只能得到类型,不能用来定义变量)

4.右值引用

左值:可以取地址的表达式,因为可以取地址因此可以给其赋值,所以可以出现"="的左右任意一侧。对左值进行引用,就是给其取个别名,称为左值引用。

右值:不可以取地址的表达式,无法给其赋值,因此只能出现在"="右侧,对右值进行引用,称为右值引用。常见的右值有表达式返回值、常量、函数返回值(返回非左值引用的类型才行)。

需注意,右值不能取地址,无法给其赋值,但是,右值引用后会有一份地址保存这个值,就可以通过这个地址对其进行修改。若不希望被修改,就用const修饰右值引用。

左值引用与右值引用区别:

左值引用:正常只能引用左值,用const修饰可以引用左值和右值

右值引用:正常只能引用右值,但是给左值加move可以被右值引用

int i=1;
int &&a=move(i);  //此处i是左值,但move(i)返回的是右值

右值引用应用:

常用于函数返回临时对象时,此时无法返回左值引用,因为变量是出了函数作用域就销毁的。若不用右值引用,则正常需要进行两次深拷贝(编译器可能优化成一次),但右值引用可以不进行深拷贝,极大提高了效率。

而右值引用的原理是,认为临时对象出了作用域就要被销毁,因此可以直接拿过来临时对象的成员,这样就不用进行深拷贝了。这就是移动构造的思想。

同时,也有移动赋值,函数参数是右值引用,原理也是直接交换对象的成员,而非深拷贝。

5.完美转发:

用于有万能模版的情况下。而万能模版就是一种特殊的模版函数,其成员是T&&val,此时可以接受左值引用也可以接受右值引用,因此是万能模版。而无论是左值引用还是右值引用,其类型都是左值(注:是右值引用是左值,不是右值是左值!),因此,若用右值引用做参数,只能调用以左值引用为形参的函数,无法调用以右值为形参的函数,若想要调用只能用move,但这样就只能调用右值引用为参数的。为了实现左值引用可以调用左值为参数的函数,右值引用可以调用右值为参数的函数,在引用前加一个forward,效果是右值引用则返回一个右值,左值引用就返回一个左值。

6.默认成员函数的增加:

增加了默认移动构造和默认移动赋值构造,编译器自己写默认移动构造函数的前提是:

1.用户自己未编写移动构造

2.用户为编写拷贝构造、赋值构造和析构函数。

使用:对于内置类型进行逐字节拷贝,对于自定义类型若有移动构造调用移动构造,若没有调用拷贝构造。

(对于默认移动赋值前提和使用极其类型,就是把移动构造改成移动赋值即可)

7.强制生成/销毁默认成员函数:

强制生成:default;    强制销毁:delete    一般称=delete的函数是删除函数

class bb
{
   public:
   
   bb()=default;//强制生成
   
   bb(const bb&b)=delete;//强制删除

}

8.可变参数模版:

写法:

template<class...Args>

返回类型 函数名(Args...args)

//函数体

}

args前面有...,是可变模版参数。称前面有...的参数为参数包,参数包里包含0到N个可变模版参数。

如何获取参数包里的可变模版参数:

1.用递归展开:

void _get()
{
	cout<< endl;
}

template<class T,class...Args>
void _get(T& val,Args...args)
{
	cout << val << " ";
	_get(args...);
}

template<class...Args>
void get(Args...args)
{
	_get(args...);
}

int main()
{
	get(1, 2.2, "xxx");
}

原理就是不断用T获取最左的那个可变模版参数,然后不断缩小参数包内的可变模版参数的个数,最后当可变模版参数为0时调用打印空格,实现获取所有的参数。

2.利用逗号表达式:

template<class T>
void print(T& val)
{
	cout<< val << " ";
}

template<class...Args>
void get(Args...args)
{
	//_get(args...);

	int a[] = { (print(args),0)... };
	cout << endl;
}

原理是利用初始化列表展开成(print(args1),0)、(print(args2),0)...,同时利用逗号表达式依次去执行每个()内的两条表达式,实现获取每个参数。

9.lambda表达式

形式:

[]+()+mutable+->+{};

其中:

[]:捕捉列表,包含在于lamabda相同作用域内的变量.[var]表示捕获变量var。[=]表示捕获父作用域内所有变量。[&var]表示捕获var变量的引用。[&=]表示父作用域所有变量的引用。[this]表示捕获当前的this指针。[]内可以包含多钟类型的捕捉,比如[&var1,var2,this],但注意相同变量不能重复捕获。

():函数参数列表,若无参数可以省略

mutable:若无mutable则表示该lambda函数是const修饰的函数,若有则可以取消常量性。注:若加上mutable,则参数列表无论有无参数都得写"()"。

->:指向返回类型,若返回类型是空可以省略,若返回类型十分明确也可以省略让编译器自动推导。

{}:函数体部分。

ps:只有[]和{}部分一定不能省略

lambda的底层代码是仿函数,若想要用一个变量接受该表达式,则需要用auto修饰变量,因为lambda返回的类型由编译器自己决定,无法显示知道并调用。

10.包装器:

头文件是<functional>

function<返回类型(函数参数)>对象={}。

作用:可以让对象赋值为仿函数、函数形参、lambda函数(但返回类型和函数参数必须相同)

(注:当接受的是lambda函数时,不管[],只要()和返回类型一致就行)

11.bind:

类型一个函数模版。功能是接受一个可调用对象,返回一个新的可调用对象去适应原本的函数参数,如修改函数参数的先后顺序和个数。

使用:

auto newcallable=bind(callable,arg_list),其中newcallable是新生成的可调用对象,callable是原可调用对象,arg_list是一个逗号表达式,包含callable的参数。

arg_list:可以是placeholder::_x,表示当前位置对应函数第x个参数。

如上图,newf中1对应的是placeholder::_2,故对应lambda的第二个函数参数b。

arg_list也可以是一个固定变量或常量,表示该位置是一个固定的函数参数,一般用于某个调用的参数固定不变的情况。

如上图,第一个参数a对应的位置为常量111,此时调用时就不需要在显示写该参数了,因此newf调用时只写了一个参数。

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值