C++11 Lambda表达式(匿名函数)详解
使用 STL 時,往往會大量用到函數對象,為此要編寫很多函數對象類。有的函數對象類只用來定義了一個對象,而且這個對象也只使用了一次,編寫這樣的函數對象類就有點浪費。
而且,定義函數對象類的地方和使用函數對象的地方可能相隔較遠,看到函數對象,想要查看其 operator() 成員函數到底是做什么的也會比較麻煩。
對于只使用一次的函數對象類,能否直接在使用它的地方定義呢?Lambda 表達式能夠解決這個問題。使用 Lambda 表達式可以減少程序中函數對象類的數量,使得程序更加優雅。
Lambda 表達式的定義形式如下:
[外部變量訪問方式說明符] (參數表) -> 返回值類型 {語句塊 }其中,“外部變量訪問方式說明符”可以是=或&,表示{}中用到的、定義在{}外面的變量在{}中是否允許被改變。=表示不允許,&表示允許。當然,在{}中也可以不使用定義在外面的變量。“-> 返回值類型”可以省略。
下面是一個合法的Lambda表達式:
[=] (int x, int y) -> bool {return x%10 < y%10; }Lambda 表達式實際上是一個函數,只是它沒有名字。下面的程序段使用了上面的 Lambda 表達式:
int a[4] = {11, 2, 33, 4}; sort(a, a+4, [=](int x, int y) -> bool { return x%10 < y%10; } ); for_each(a, a+4, [=](int x) { cout << x << " ";} );這段程的輸出結果是:
11 2 33 4
程序第 2 行使得數組 a 按個位數從小到大排序。具體的原理是:sort 在執行過程中,需要判斷兩個元素 x、y 的大小時,會以 x、y 作為參數,調用 Lambda 表達式所代表的函數,并根據返回值來判斷 x、y 的大小。這樣,就不用專門編寫一個函數對象類了。
第 3 行,for_each 的第 3 個參數是一個 Lambda 表達式。for_each 執行過程中會依次以每個元素作為參數調用它,因此每個元素都被輸出。
下面是用到了外部變量的Lambda表達式的程序:
#include <iostream> #include <algorithm> using namespace std; int main() {int a[4] = { 1, 2, 3, 4 };int total = 0;for_each(a, a + 4, [&](int & x) { total += x; x *= 2; });cout << total << endl; //輸出 10for_each(a, a + 4, [=](int x) { cout << x << " "; });return 0; }程序的輸出結果如下:
10
2 4 6 8
第 8 行,[&]表示該 Lambda 表達式中用到的外部變量 total 是傳引用的,其值可以在表達式執行過程中被改變(如果使用[=],編譯無法通過)。該 Lambda 表達式每次被 for_each 執行時,都將 a 中的一個元素累加到 total 上,然后將該元素加倍。
實際上,“外部變量訪問方式說明符”還可以有更加復雜和靈活的用法。例如:
- [=, &x, &y]表示外部變量 x、y 的值可以被修改,其余外部變量不能被修改;
- [&, x, y]表示除 x、y 以外的外部變量,值都可以被修改。
例如下面的程序:
#include <iostream> using namespace std; int main() { int x = 100,y=200,z=300;auto ff = [=,&y,&z](int n) {cout <<x << endl;y++; z++;return n*n;};cout << ff(15) << endl;cout << y << "," << z << endl; }程序的輸出結果如下:
100
225
201, 301
第 6 行定義了一個變量 ff,ff 的類型是 auto,表示由編譯器自動判斷其類型(這也是 C++11 的新特性)。本行將一個 Lambda 表達式賦值給 ff,以后就可以通過 ff 來調用該 Lambda 表達式了。
第 11 行通過 ff,以 15 作為參數 n 調用上面的 Lambda 表達式。該 Lambda 表達式指明,對于外部變量 y、z,可以修改其值;對于其他外部變量,例如 x,不能修改其值。因此在該表達式執行時,可以修改外部變量 y、z 的值,但如果出現試圖修改 x 值的語句,就會編譯出錯。
原文地址:http://c.biancheng.net/view/433.html
總結
以上是生活随笔為你收集整理的C++11 Lambda表达式(匿名函数)详解的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: C++11 auto和decltype关
- 下一篇: Qt Remote Object(QtR