淺析C++11新特性的Lambda運算式

來源:互聯網
上載者:User
lambda簡介

熟悉Python的程式員應該對lambda不陌生。簡單來說,lambda就是一個匿名的可調用代碼塊。在C++11新標準中,lambda具有如下格式:

[capture list] (parameter list) -> return type { function body }

可以看到,他有四個組成部分:

1.capture list: 捕獲列表

2.parameter list: 參數列表

3.return type: 傳回型別

4.function body: 執行代碼

其中,參數列表和傳回型別可以忽略。

下面,具體看幾個簡單的例子:

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被調用時,此引用仍然有效。

捕獲列表還可以採用隱式捕獲的方式,即讓編譯器通過lambda的執行代碼來判斷需要捕獲哪些局部變數。

隱式捕獲可以捕獲值、引用或者兩者混合:

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關鍵字。否則會有編譯錯誤。

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中的值,在第一次調用之後,就被變成了定位字元Tab;但是在lambda之外,space仍然是空格。

傳回型別

lambda的傳回型別採用尾置傳回型別的方式。一般的:

1.lambda如果只包含return語句,則編譯器可以推斷其傳回型別,此時可以不顯示指定傳回型別;

2.否則,編譯器假定lambda返回void,而返回void的函數不可以反悔任何具體值,這在大多數情況下是個矛盾,因此需要顯示指定傳回型別。

但是,經過實際測試,目前的g++編譯器更聰明了:對於第2點,目前只要編譯器可以從lambda函數體中推斷出函數的傳回型別,就不需要顯式指定傳回型別,例如:

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語句,但是編譯器可以根據return語句推斷出,其傳回值應該是一個int類型,所以可以省略尾置傳回型別。

但是,像下面這種形式,由於編譯器在推斷傳回型別時發現了不一致,所以必須顯式的指定傳回型別:

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;

總結

1.lambda運算式形式: [capture list] (parameter list) -> return type { function body },其中parameter list和return type可以省略。

2.捕獲列表可以捕獲值[val],也可以捕獲引用[&ref]。

3.捕獲列表還可以隱式捕獲局部變數,同樣有捕獲值[=]和捕獲引用[&]兩種方式,初次之外還可以混合捕獲[&, val]或者[=, &ref]。

4.當lambda需要修改捕獲的值時,需要加上mutable關鍵字。

4.當lambda無法自動推斷出傳回值類型時,需要通過尾置傳回型別的方式顯示指定。

以上就是C++11新特性之Lambda運算式的全部內容,希望本文對大家學習C++有所協助。

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.