C++11

目录

1.可变参数模板

1.1概念

1.2参数包的展开

2.stl的emplace接口

3.lambda表达式

lambda表达式形式

demo代码

4.function包装器

5.bind包装器


1.可变参数模板

1.1概念

有了可变参数模板,可以创建任意参数的函数模板和类模板。

cpp 复制代码
template <class ...Args> // ...的位置,怎么规定就怎么用吧。。。。。。
int add(Args... arg)
{
	return 0;
}

int main()
{
	vector<int> v;
	string s;
	add(1, 3, 4, 5, 6);
	add(1,'c', v,s);
	return 0;
}

上述代码,直接生成解决方案是没有任何问题的。

1.2参数包的展开

cpp 复制代码
template<class T>
void show(T t)
{
	cout << t << endl;
}

template<class T, class ...Args>
void show(T t,Args... arg)
{
	cout << t << endl;
	show(arg...);
}

int main()
{
	vector<int> v;
	string s;
	show(1, 3, 4, 5, 6);
	cout << endl;
	return 0;
}

这个也不知道是哪个神人想出来的,大概思路就是,[1,3,4,5,6],第一次调用show将[1]传给t,[3,4,5,6]传给arg,第二次递归将[3]传给t,[4,5,6]传给arg,依次类推就将所有参数解包。

2.stl的emplace接口

这个也是c++11之后新添加的功能,他支持万能引用,指出可变参数

插入一个新的元素在vector的末尾元素的后面,这个新的元素是使用args为参数构造的。

cpp 复制代码
class date
{
public:
	date(int year = 2000, int month = 1, int day = 1)
		:_year(year)
		,_month(month)
		,_day(day)
	{
		cout<< "date(int year = 2000, int month = 1, int day = 1)" << endl;
	}

private:
	int _year;
	int _month;
	int _day;
};

int main()
{
	vector<date> v;
	v.push_back({1999,3,3});
	//v.push_back(1999, 3, 3);错误做法
	v.emplace_back(1999,3,3);
	return 0;
}

v.push_back({1999,3,3});就是隐式类型转换,构造出一个date临时对象,在拷贝临时对象放入vector中。

v.emplace_back(1999,3,3);直接构造一个对象,放入vector中。

emplace接口对浅拷贝的类更有意义的,对于浅拷贝emplace可以减少一次拷贝构造。

但是对于深拷贝有移动构造的类,emplace减少一次移动构造,移动构造代价很小,效率提升不明显。

3.lambda****表达式

在c++98中,我们使用sort这类的接口,想要比叫大于或者是小于,必须自己写一个仿函数,非常的不人性,非常的麻烦。

cpp 复制代码
template<class T>
class compare
{
public:
	bool operator()(T left,T right)
	{
		return left > right;
	}
};

int main()
{
	compare<int> cmp;
	vector<int>a = {1,2,4,5,6};
	sort(a.begin(),a.end(),cmp);
	for (auto num : a)
	{
		cout << num << endl;
	}
	return 0;
}

在c++11中我们可以使用lambda表达式,代替仿函数。

lambda表达式形式

[ capture-list] (parameters) mutable -> return-type { statement }

**capture-list:**捕捉列表,用于捕捉参数,捕捉的参数是原参数的拷贝

[var]:表示值传递方式捕捉变量var

[=]:表示值传递方式捕获所有父作用域中的变量(包括this)
[&var]:表示引用传递捕捉变量var
[&]:表示引用传递捕捉所有父作用域中的变量(包括this)
[this]:表示值传递方式捕捉当前的this指针

**(parameters):**参数列表,和函数的参数列表一样。

**mutable:**关键字,lambda默认是const,mutbable可以取消const。

**return-type:**返回值值,可写可不写,lambda表达式会自动推导。

**{ statement }:**函数体

demo代码

cpp 复制代码
class goods
{
public:
	goods(string name = "未知", int price = 0)
		:_name(name)
		,_price(price)
	{}
	string _name;
	int _price;
};

int main()
{
	vector<goods> v = { {"苹果",3},{"香蕉",6},{"栗子",1} };
	sort(v.begin(), v.end(), [](goods& g1, goods& g2)->bool {return g1._price > g2._price; });
	for(auto g : v)
	{
		cout << g._name << endl;
	}
	return 0;
}

lamdba表达式的底层还是仿函数的,lamdba表达式是可以拷贝,但是不能赋值。

cpp 复制代码
int  main()
{
	auto lamdba1 = [] {cout << "hello word" << endl; };
	auto lamdba2 = [] {cout << "hello word" << endl; };
	lamdba1 = lamdba2;

	return 0;
}

错误原因就是他们两个类型不同。

4.function包装器

包装器是一个类模板,它可以用来接受,函数指针,函数对象,lamdba表达式。

cpp 复制代码
#include<functional>
// 类模板原型如下
template <class T> function;     // undefined
template <class Ret, class... Args>
class function<Ret(Args...)>;

Ret是返回值,Args是函数参数。

demo代码

cpp 复制代码
int add(int a,int b)
{
	return a + b;
}

class cadd
{
public:
	int operator()(int a, int b)
	{
		return a + b;
	}
};
int  main()
{
	function<int(int, int)> func1 = add;
	cout<<func1(1,3)<<endl;
	function<int(int, int)> func2 = cadd();
	cout << func2(2, 4)<<endl;
	function<int(int, int)> func3 = [](int a, int b)->int { return a + b; };
	cout << func3(2, 4) << endl;
	return 0;
}

5.bind包装器

bind是一个函数模板,它就像一个函数包装器(适配器),接受一个可调用对象(callable object),生成一个新的可调用对象来"适应"原对象的参数列表 。
主要就是有两个功能,改变参数位置,绑定参数。

cpp 复制代码
template <class Ret, class Fn, class... Args>
bind (Fn&& fn, Args&&... args);

参数fn接收一个函数,Args就是函数的参数。

_1代表的就是函数第一个参数,_1是在placeholders这个类域里的。

参数绑定

绑定的位置,与函数参数位置对应。

cpp 复制代码
auto func3 = bind(sub,1,5);
cout << func3 () << endl;

成员函数的绑定

成员函数别忘记,this,每个成员函数都有一个this指针。

cpp 复制代码
auto func4 = bind(&csub::sub, csub(), placeholders::_1, placeholders::_2);
cout << func4(1,7) << endl;
相关推荐
Song10 分钟前
C++:求梯形面积
开发语言·c++
十年一梦实验室1 小时前
【C++】相机标定源码笔记- RGB 相机与 ToF 深度传感器校准类
开发语言·c++·笔记·数码相机·计算机视觉
蜉蝣之翼❉1 小时前
c++ 简单线程池
开发语言·c++
夏天的阳光吖2 小时前
每日一题---OJ题:分隔链表
数据结构·链表
shuai132_2 小时前
关于std::memory_order_consume
开发语言·c++
CylMK3 小时前
NOI大纲——普及组——二叉搜索树
开发语言·数据结构·c++·算法
to-and-fro3 小时前
Nginx中封装的数据结构
运维·数据结构·nginx
3 小时前
数据结构:期末考 第六次测试(总复习)
c语言·数据结构
hummhumm3 小时前
数据结构第06节:栈
java·开发语言·数据结构·spring boot·spring·java-ee·intellij-idea
Smark.3 小时前
06.01_111期_C++_unordermap_unorderset
开发语言·c++