1, 定義
lambda 表達(dá)式就是一個(gè)函數(shù)(匿名函數(shù)),,也就是一個(gè)沒有函數(shù)名的函數(shù),。為什么不需要函數(shù)名呢,因?yàn)?strong>我們直接(一次性的)用它,,嵌入式用的它,,不需要其他地方調(diào)用它。 lambda 表達(dá)式也叫閉包,。閉就是封閉的意思(封閉就是其他地方都不調(diào)用它),,包就是函數(shù)。
lambda 表達(dá)式其實(shí)就是一個(gè)函數(shù)對(duì)象,,他內(nèi)部創(chuàng)建了一個(gè)重載()操作符的類。
lambda 表達(dá)式的簡(jiǎn)單語法如下:[capture] (parameters) -> return value { body }, 只有 [capture] 捕獲列表和 { body } 函數(shù)體是必選的,,其他可選,。
2, 最簡(jiǎn)單的一個(gè) lambda 表達(dá)式(調(diào)用) int main() { [] {}();//三部分,[] : 代表lambda表達(dá)式的開始;{} : 代表函數(shù)體,,函數(shù)體里面什么都沒有;() : 代表函數(shù)調(diào)用. }
等價(jià)于: void f() {
} int main() { f(); }
像其他函數(shù)一樣,,我們需要一個(gè)參數(shù)列表:()。所以: [] {}(); 也可以寫為:
第二個(gè)位置加了一個(gè)()代表函數(shù)參數(shù),。如果什么參數(shù)都沒有,,就可以省略 ()。 3, 再加點(diǎn)打印的東西: #include <iostream> using namespace std; int main() { [] { cout << 'Hello, World!'<<endl; }(); }
也可以這樣: #include <iostream> using namespace std; int main() { auto lam = [] { cout << 'Hello, World!'<<endl; }; lam(); }
4, 返回值
-> int :代表此匿名函數(shù)返回 int,。大多數(shù)情況下lambda表達(dá)式的返回值可由編譯器猜測(cè)得出,,因此不需要我們指定返回值類型。 int main() { auto lam =[]() -> int { cout << 'Hello, World!'; return 88; }; //auto lam =[]() { cout << 'Hello, World!'; return 88; };//自動(dòng)推導(dǎo)返回值 auto ret = lam(); cout<<ret<<endl;//輸出88 auto lam2 =[]() -> string { cout << 'Hello, World!'; return 'test'; }; auto ret1 = lam2(); cout<<ret1<<endl;//輸出test }
5, 捕捉變量 變量捕獲才是成就 lambda 卓越的秘方,。
[] 不捕獲任何變量, 這種情況下lambda表達(dá)式內(nèi)部不能訪問外部的變量,。 [&] 以引用方式捕獲所有變量 [=] 用值的方式捕獲所有變量(可能被編譯器優(yōu)化為const &) [=, &foo] 以引用捕獲變量foo, 但其余變量都靠值捕獲 [&, foo] 以值捕獲foo, 但其余變量都靠引用捕獲 [bar] 以值方式捕獲bar; 不捕獲其它變量 [this] 捕獲所在類的this指針 (Qt中使用很多,如此lambda可以通過this訪問界面控件的數(shù)據(jù))
int a=1,b=2,c=3; auto lam2 = [&,a](){//b,,c以引用捕獲,,a以值捕獲,。 b=5;c=6;//a =1; a不能賦值 cout << a<<b<<c<<endl;//輸出 1 5 6 }; lam2();
6, 毋庸質(zhì)疑,lambda最大的一個(gè)優(yōu)勢(shì)是在使用STL中的算法 (algorithms) 庫時(shí): vector<string> address{'111','222',',333','.org','wwwtest.org'}; for_each(address.begin(),address.end(),[](const string& str){cout<<str<<endl;});
如此一行代碼就可以循環(huán)打印容器的數(shù)據(jù),。 再看一個(gè)例子,,以前數(shù)組排序代碼(第二代排序,第一代是自己寫)是這樣的:
int arr[] = {6,4,3,2,1,5}; bool compare(int& a,int& b)//謂詞函數(shù) { return a>b; } std::sort(arr, arr+6, compare);
現(xiàn)在: std::sort(arr, arr+6, [](const int& a,const int& b){return a>b;});//降序排序 //std::sort(arr, arr+6, [](const auto& a,const auto& b){return a>b;}); //C++14支持基于類型推斷的泛型lambda表達(dá)式,。 std::for_each(begin(arr),end(arr),[](const int& e){cout<<'After:'<<e<<endl;});//6,5,4,3,2,1
lambda表達(dá)式取代了謂詞函數(shù),,一行代碼搞定排序,再一行代碼搞定打印數(shù)組所有數(shù)據(jù),。 7, 打印類型 auto lam = [](){cout<<'hi'<<endl;}; cout<<typeid(lam).name()<<endl;//輸出 class <lambda_6c3f764f39072358acd689d114b4c204> int i1; cout<<'typeid:'<<typeid(i1).name()<<endl;//輸出 i cout<<'typeid:'<<typeid(float).name()<<endl;//輸出 f
8,
創(chuàng)建lambda函數(shù)的一個(gè)原因是有些人創(chuàng)建了一個(gè)希望接受(lambda函數(shù))參數(shù)的函數(shù),。 lambda 的引入給我們帶來了一種全新的編程體驗(yàn),它可以讓我們把 “function” 當(dāng)做是 “data” 一樣傳遞,,并且使我們從繁瑣的語法中解放出來,,更加關(guān)注于 “算法” 本身。 新的 std::function 是傳遞lambda函數(shù)的最好的方式,,不管是傳遞參數(shù)還是返回值,。
以下代碼將lambda表達(dá)式作為函數(shù)參數(shù)傳遞。程序的作用很簡(jiǎn)單,,是從一個(gè)地址簿中查找滿足條件的地址(匹配字符串或長度等規(guī)則),。( 參考: 「C++11」Lambda 表達(dá)式)
#include <iostream> #include <string> #include <vector> #include <functional> #include <algorithm> using namespace std;
class AddressBook { public: //提供一個(gè)通用的查找方法,以供查詢(匹配的地址),,這個(gè)方法接受一個(gè)查找規(guī)則的函數(shù)作為參數(shù) std::vector<string> findMatchingAddresses (std::function<bool (const string&)> func) { std::vector<string> results; for ( auto it = _addresses.begin(), end = _addresses.end(); it != end; ++it ){ // 調(diào)用傳遞到findMatchingAddresses的函數(shù)并檢測(cè)是否匹配規(guī)則 if ( func( *it ) ){ results.push_back( *it ); } } return results; } void SetAddress(const std::vector<std::string> &address) { _addresses = address; } private: std::vector<string> _addresses; }; //聲明一個(gè)全局的AddressBook對(duì)象 AddressBook global_address_book; //查找匹配名字的地址 vector<string> findAddressesFromName (const string &name) { return global_address_book.findMatchingAddresses( [&] (const string& addr) { return addr.find(name) != string::npos; } ); } //查找匹配長度的地址(>min_len) vector<string> findAddressesLen (const size_t &min_len) { return global_address_book.findMatchingAddresses( [&] (const string& addr) { return addr.length() >= min_len; } ); }
int main() { //初始化AddressBook對(duì)象的成員 _addresses vector<string> address{'China chengdu','China hunan','TaiWan taibei','American alasijia','Japan dongjing'}; global_address_book.SetAddress(address); //查找包含 China 的地址 auto ret = findAddressesFromName('China'); for_each(ret.begin(),ret.end(),[](string &i){cout<<i<<' ';});cout<<endl;//輸出: China chengdu 和 China hunan //查找長度大于 15 的地址 auto ret2 = findAddressesLen(15); for_each(ret2.begin(),ret2.end(),[](string &i){cout<<i<<' ';});cout<<endl;//輸出: American alasijia
return 0; }
完,。 版權(quán)聲明:本文為CSDN博主「宇宙379」的原創(chuàng)文章,遵循CC 4.0 BY-SA版權(quán)協(xié)議,,轉(zhuǎn)載請(qǐng)附上原文出處鏈接及本聲明,。
|