先来看个问题:在一个string的vector容器中,寻找第一个长度大于4的元素。
inline bool check_size(const string& s) { return s.size() > 4; } int main(int argc, char *argv[]) { QCoreApplication a(argc, argv); vector<string> v{"1", "22", "333", "4444", "55555", "666666", "7777777"}; cout << *find_if(v.begin(),v.end(),check_size) << endl; return a.exec(); }
很简单,只要使用find_if和一个函数就可以。可是假如我们之前不确定界定元素大小的临界值,这可怎么办呢?这是就可以使用“lambda”表达式。它的出现刚好可以弥补类型find_if这些泛型算法只能使用一元谓词的尴尬。
int main(int argc, char *argv[]) { QCoreApplication a(argc, argv); vector<string> v{"1", "22", "333", "4444", "55555", "666666", "7777777"}; string::size_type sz = 4; auto c = find_if(v.begin(),v.end(),[sz](const string &s) {return s.size() > sz;}); cout << *c << endl; return a.exec(); }
假如我们想要直接把临界值之前的值打印输出,“lambda”可以捕获一个ostream的引用。
int main(int argc, char *argv[]) { QCoreApplication a(argc, argv); vector<string> v{"1", "22", "333", "4444", "55555", "666666", "7777777"}; string::size_type sz = 4; auto &os = cout; auto c = find_if(v.begin(),v.end(),[&os,sz](const string &s) { os << s << " "; return s.size() > sz;}); cout << " the first item over sz : " << *c << endl; return a.exec(); }
我们发现,当“lambda”捕获的元素多了之后,表达式就显的很臃肿,这时候回觉得还是用函数来的比较方便,那有没有可以解决函数长度参数问题的方法呢?
这里有个方法就是使用名为bind的标准库函数。
inline bool check_size(const string& s,string::size_type sz, ostream &os) { os << s << " "; return s.size() > sz; } int main(int argc, char *argv[]) { QCoreApplication a(argc, argv); vector<string> v{"1", "22", "333", "4444", "55555", "666666", "7777777"}; string::size_type sz = 4; auto &os = cout; auto c = find_if(v.begin(),v.end(),bind(check_size,_1,sz,os)); cout << " the first item over sz : " << *c << endl; return a.exec(); }
可是编译回报错,原因在于bind不能拷贝一个ostream参数,这时候可以使用标准库中的ref函数
inline bool check_size(const string& s,string::size_type sz, ostream &os) { os << s << " "; return s.size() > sz; } int main(int argc, char *argv[]) { QCoreApplication a(argc, argv); vector<string> v{"1", "22", "333", "4444", "55555", "666666", "7777777"}; string::size_type sz = 4; auto &os = cout; auto c = find_if(v.begin(),v.end(),bind(check_size,_1,sz,ref(os))); cout << " the first item over sz : " << *c << endl; return a.exec(); }
这样就可以解决传递长度参数的问题。