C++ lambda表示式入門
阿新 • • 發佈:2018-12-19
1.lambda表示式 lambda表示式 是一個函式,一個匿名函式,也就是沒有函式名的函式,為什麼不需要函式名呢,因為我們直接(一次性的)用它,嵌入式用的它,不需要其他地方用它。 也叫閉包,閉就是封閉的意思,就是其他地方都不用他,包就是函式。 lambda表示式 其實就是一個函式物件,他內部建立了一個過載()操作符的類。
最簡單的一個lambda表示式。
int main()
{
[] {}();//[]代表lambda表示式的開始,{}代表函式體,什麼都沒有,()代表呼叫函式.
}
等價與:
void f()
{
}
int main()
{
f();
}
像其他函式一樣,我們需要一個引數列表:()。所以: [] {}(); 也可以寫:
[]() {}();
加了一個()代表函式引數。什麼引數都沒有,就可以省略。 3. 再加點列印的東西:
int main()
{
[] { cout << "Hello, World!"; }();
}
也可以這樣:
int main()
{
auto lam = [] { cout << "Hello, World!"; };
lam();
}
- 返回值 -> int :代表返回int。
int main() { auto lam =[]() -> int { cout << "Hello, World!"; return 1; }; auto ret = lam(); auto lam2 =[]() -> string { cout << "Hello, World!"; return "test"; }; auto ret1 = lam2(); }
- 捕捉變數 捕捉變數是變數捕獲才是成就lambda卓越的祕方。
[] 不捕獲任何變數 [&] 以引用方式捕獲所有變數 [=] 用值的方式捕獲所有變數(可能被編譯器優化為const &) [=, &foo] 以引用捕獲foo, 但其餘變數都靠值捕獲 [&, foo] 以值捕獲foo, 但其餘變數都靠引用捕獲 [bar] 以值方式捕獲bar; 不捕獲其它變數 [this] 捕獲所在類的this指標
int a=1,b=2,c=3; auto lam2 =[&,a](){ cout << a<<b<<c<<endl;};//b,c以引用捕獲,a以值捕獲。 lam2();
毋庸質疑,lambda最大的一個優勢是在使用STL中的演算法(algorithms)庫時。
vector<string> address{"111","222",",333",".org","wwwtest.org"};
for_each(address.begin(),address.end(),[](string& str){cout<<str<<endl;});
如此一行程式碼就可以迴圈列印容器的資料。
建立lambda函式的一個原因是有些人建立了一個希望接受lambda函式的函式。 新的std::function是傳遞lambda函式的最好的方式,不管是傳遞引數還是返回值。 ( 參考: https://www.cnblogs.com/kedebug/p/3224561.html)
#include <iostream>
#include <string>
#include <vector>
#include <functional>
#include <algorithm>
using namespace std;
class AddressBook
{
public:
std::vector<string> findMatchingAddresses (std::function<bool (const string&)> func)
{
std::vector<string> results;
for ( auto itr = _addresses.begin(), end = _addresses.end(); itr != end; ++itr )
{
// 呼叫傳遞到findMatchingAddresses的函式並檢測是否匹配規則
if ( func( *itr ) )
{
results.push_back( *itr );
}
}
return results;
}
void SetAddress(const std::vector<std::string> &address)
{
_addresses = address;
}
private:
std::vector<string> _addresses;
};
AddressBook global_address_book;
//查詢匹配名字的地址
vector<string> findAddressesFromName (const string &name)
{
return global_address_book.findMatchingAddresses(
[&] (const string& addr) { return addr.find(name) != string::npos; }
);
}
//查詢匹配長度的地址
vector<string> findAddressesLen (const int &min_len)
{
return global_address_book.findMatchingAddresses( [&] (const string& addr) { return addr.length() >= min_len; } );
}
int main()
{
vector<string> address{"china chengdu","china hunan","taiwan taibei","american alaisjia","riben dongjing"};
global_address_book.SetAddress(address);
auto ret = findAddressesFromName("china");
for_each(ret.begin(),ret.end(),[](string &i){cout<<i<<",";});cout<<endl;
auto ret2 = findAddressesLen(15);
for_each(ret2.begin(),ret2.end(),[](string &i){cout<<i<<",";});cout<<endl;
return 0;
}