亲宝软件园·资讯

展开

C++lambda表达式

配的上了吗 人气:0

例:

stable_sort(words.begin(), words.end(), 
		[](const string& s1, const string& s2) 
		{ return s1.size() < s2.size(); });
class ShorterString 
{
public:
	bool operator()(const string& s1,const string& s2)const
	{
		return s1.size() < s2.size();
	}
};
// 等价于
stable_sort(words.begin(), words.end(), ShorterString();
// 或者
ShorterString s;
stable_sort(words.begin(), words.end(), s;

写一个lambda表达式之后,编译器将该表达式翻译为一个未命名类的未命名对象。

且lambda表达式产生的类中含有一个重载的函数调用运算符。且该函数调用运算符成员的形参列表和函数体与lambda表达式完全一样。

此时,stable_sort的内部代码每次比较两个string时就会"调用"这一对象。此对象将会调用运算符的函数体,返回true or false

我们知道,lambda表达式可以捕获变量,且分为引用捕获和值捕获,而当引用捕获时,由程序负责确保lambda执行时引用所引的对象确实存在,而编译器就会直接使用该引用,无需在lambda产生的类中将其存储为数据成员

根据上一段话的介绍,大致也可以猜想出,若采用值捕获的方式,被值捕获的变量会被拷贝至lambda中。因此,这种lambda产生的类必须为每个值捕获的变量创建对应的数据成员。 同时创建构造函数。令其使用捕获的变量的值来初始化数据成员。

默认情况下lambda表达式不可以改变它捕获的变量。所以重载函数调用运算符的函数默认情况下会被定义为const。

举例:

auto wc = find_if(words.begin(), words.end(), 
    [sz] (const string& s) { return s.size() >= sz; } );
class Sizecomp
{
public:
	Sizecomp(size_t n) :sz(n) {}
	bool operator()(const string& s)const
	{
		return s.size() >= sz;
	}
private:
	size_t sz;
};
// 等价于
auto wc = find_if(words.begin(), words.end(), Sizecomp(sz));

此类无默认构造函数,因此要想使用这个类必须提供一个实参。数据成员sz对应通过值捕获的变量。用此值捕获变量的值来初始化该sz数据成员。

在C++11中, lambda是通过匿名的函数对象来实现的,因此我们可以把lambda看作是对函数对象在使用方式上进行的简化。

当代码需要一个简单的函数,并且这个函数并不会在其他地方被使用时,就可以使用lambda来实现,此时它所起的作用类似于匿名函数。 但如果这个函数需要多次使用,并且它需要保存某些状态的话,使用函数对象则更合适一些。

练习:编写程序,统计一个vector<string>中元素size处于1-9的有几个,大于等于10的有几个

class StrLenBetween
{
public:
	StrLenBetween(int m, int x) :min(m), max(x) {}
	bool operator()(const string& s)const
	{
		return s.size() >= min && s.size() <= max;
	}
private:
	int min;
	int max;
};
void readStr(istream& is, vector<string>& v)
{
	string s;
	while (is >> s)
		v.push_back(s);
}
int main()
{
	vector<string> vec;
	readStr(cin, vec);
	const int min = 1;
	const int max = 10;
	cout << "len 1~9 : " << count_if(vec.begin(), vec.end(), [min, max](const string& s)->bool {return s.size() >= 1 && s.size() <= 9; }) << endl;
	cout << "len >= 10 : " << count_if(vec.begin(), vec.end(), [max](const string& s)->bool {return s.size() >= max; }) << endl;
	cout << "len 1~9 : " << count_if(vec.begin(), vec.end(), StrLenBetween(min, max)) << endl;
	return 0;
}

主函数第一个cout和第三个cout意义相同。

加载全部内容

相关教程
猜你喜欢
用户评论