淺析C++11新特性的Lambda表達式
lambda簡介
熟悉Python的程序員應該對lambda不陌生。簡單來說,lambda就是一個匿名的可調(diào)用代碼塊。在C++11新標準中,lambda具有如下格式:
[capture list] (parameter list) -> return type { function body }
可以看到,他有四個組成部分:
1.capture list: 捕獲列表
2.parameter list: 參數(shù)列表
3.return type: 返回類型
4.function body: 執(zhí)行代碼
其中,參數(shù)列表和返回類型可以忽略。
下面,具體看幾個簡單的例子:
auto f1 = [] { return 1; }; auto f2 = [] () { return 2; }; cout<<f1()<<'\t'<<f2()<<endl;
捕獲列表
lambda中的捕獲列表既可以捕獲值,也可以捕獲引用。
捕獲值:
int test_data[] = {1, 5, 9, 7, 3, 19, 13, 17}; int border = 8; auto f3 = [border](const int &i){ if(i > border) cout<<i<<'\t'; }; for_each(begin(test_data), end(test_data), f3); cout<<endl;
捕獲引用:
auto f4 = [&border](const int &i){ if(i > border) cout<<i<<'\t'; }; border = 6; for_each(begin(test_data), end(test_data), f4); cout<<endl;
通過輸出可以看出,lambda中起作用的border是修改后的6,證實了捕獲的確是是引用。
需要注意的是,在捕獲引用時,需要保證當lambda被調(diào)用時,此引用仍然有效。
捕獲列表還可以采用隱式捕獲的方式,即讓編譯器通過lambda的執(zhí)行代碼來判斷需要捕獲哪些局部變量。
隱式捕獲可以捕獲值、引用或者兩者混合:
char space = ' '; auto f5 = [=](const int &i){ if(i > border) cout<<i<<'\t'; }; auto f6 = [&](const int &i){ if(i > border) cout<<i<<'\t'; }; auto f7 = [&, space](const int &i){ if(i > border) cout<<i<<space; }; border = 0; for_each(begin(test_data), end(test_data), f5); cout<<endl; for_each(begin(test_data), end(test_data), f6); cout<<endl; for_each(begin(test_data), end(test_data), f7); cout<<endl;
這里的f7使用的混合形式,可以讀作“除了space捕獲值之外,其他變量均捕獲引用”。
可變lambda
當lambda需要在其中修改被值捕獲的變量的值時,需要給lambda加上mutable關(guān)鍵字。否則會有編譯錯誤。
auto f8 = [&, space](const int &i) mutable { if(i > border) {cout<<i<<space; space='\t';} }; for_each(begin(test_data), end(test_data), f8); cout<<endl; cout<<1<<space<<2<<endl;
從輸出中可以看出,space在lambda f8中的值,在第一次調(diào)用之后,就被變成了制表符Tab;但是在lambda之外,space仍然是空格。
返回類型
lambda的返回類型采用尾置返回類型的方式。一般的:
1.lambda如果只包含return語句,則編譯器可以推斷其返回類型,此時可以不顯示指定返回類型;
2.否則,編譯器假定lambda返回void,而返回void的函數(shù)不可以反悔任何具體值,這在大多數(shù)情況下是個矛盾,因此需要顯示指定返回類型。
但是,經(jīng)過實際測試,目前的g++編譯器更聰明了:對于第2點,目前只要編譯器可以從lambda函數(shù)體中推斷出函數(shù)的返回類型,就不需要顯式指定返回類型,例如:
auto f9 = [](const int i){if(i % 3) return i * 3; else return i;}; transform(begin(test_data), end(test_data), begin(test_data), f9); border = 0; for_each(begin(test_data), end(test_data), f6); cout<<endl;
lambda代碼塊中有多個return語句,并且還有if/else語句,但是編譯器可以根據(jù)return語句推斷出,其返回值應該是一個int類型,所以可以省略尾置返回類型。
但是,像下面這種形式,由于編譯器在推斷返回類型時發(fā)現(xiàn)了不一致,所以必須顯式的指定返回類型:
auto f10 = [](const int i) -> double {if(i % 5) return i * 5.0; else return i;}; transform(begin(test_data), end(test_data), begin(test_data), f10); for_each(begin(test_data), end(test_data), f6); cout<<endl;
總結(jié)
1.lambda表達式形式: [capture list] (parameter list) -> return type { function body },
其中parameter list和return type可以省略。
2.捕獲列表可以捕獲值[val],
也可以捕獲引用[&ref]。
3.捕獲列表還可以隱式捕獲局部變量,同樣有捕獲值[=]和捕獲引用[&]兩種方式,初次之外還可以混合捕獲[&, val]
或者[=, &ref]
。
4.當lambda需要修改捕獲的值時,需要加上mutable關(guān)鍵字。
4.當lambda無法自動推斷出返回值類型時,需要通過尾置返回類型的方式顯示指定。
以上就是C++11新特性之Lambda表達式的全部內(nèi)容,希望本文對大家學習C++有所幫助。
上一篇:C語言數(shù)據(jù)類型轉(zhuǎn)換實例代碼
欄 目:C語言
下一篇:C 語言條件運算符詳細講解
本文標題:淺析C++11新特性的Lambda表達式
本文地址:http://mengdiqiu.com.cn/a1/Cyuyan/2145.html
您可能感興趣的文章
- 01-10淺析Linux下精確控制時間的函數(shù)
- 01-10淺析C語言中sscanf 的用法
- 01-10淺析C語言中printf(),sprintf(),scanf(),sscanf()的用法和區(qū)別
- 01-10淺析C++中memset,memcpy,strcpy的區(qū)別
- 01-10淺析C/C++中被人誤解的SIZEOF
- 01-10淺析int*p[ ]與int(*p)[ ]的區(qū)別
- 01-10淺析C語言中assert的用法
- 01-10淺析C語言頭文件和庫的一些問題
- 01-10淺析C語言中的sizeof
- 01-10淺析C++中前置聲明的應用與陷阱


閱讀排行
本欄相關(guān)
- 04-02c語言函數(shù)調(diào)用后清空內(nèi)存 c語言調(diào)用
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言的正則匹配函數(shù) c語言正則表達
- 04-02c語言用函數(shù)寫分段 用c語言表示分段
- 04-02c語言中對數(shù)函數(shù)的表達式 c語言中對
- 04-02c語言編寫函數(shù)冒泡排序 c語言冒泡排
- 04-02c語言沒有round函數(shù) round c語言
- 04-02c語言分段函數(shù)怎么求 用c語言求分段
- 04-02C語言中怎么打出三角函數(shù) c語言中怎
- 04-02c語言調(diào)用函數(shù)求fibo C語言調(diào)用函數(shù)求
隨機閱讀
- 08-05dedecms(織夢)副欄目數(shù)量限制代碼修改
- 01-10SublimeText編譯C開發(fā)環(huán)境設置
- 04-02jquery與jsp,用jquery
- 08-05織夢dedecms什么時候用欄目交叉功能?
- 01-10delphi制作wav文件的方法
- 01-11ajax實現(xiàn)頁面的局部加載
- 01-10使用C語言求解撲克牌的順子及n個骰子
- 08-05DEDE織夢data目錄下的sessions文件夾有什
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 01-10C#中split用法實例總結(jié)