C++ 11中的Lambda表達(dá)式用于定義并創(chuàng)建匿名的函數(shù)對(duì)象,以簡(jiǎn)化編程工作。Lambda的語(yǔ)法形式如下:
[函數(shù)對(duì)象參數(shù)] (操作符重載函數(shù)參數(shù)) mutable或exception聲明 ->返回值類型 {函數(shù)體} 可以看到,Lambda主要分為五個(gè)部分:[函數(shù)對(duì)象參數(shù)]、(操作符重載函數(shù)參數(shù))、mutable或exception聲明、->返回值類型、{函數(shù)體}。下面分別進(jìn)行介紹。
一、[函數(shù)對(duì)象參數(shù)],標(biāo)識(shí)一個(gè)Lambda的開(kāi)始,這部分必須存在,不能省略。函數(shù)對(duì)象參數(shù)是傳遞給編譯器自動(dòng)生成的函數(shù)對(duì)象類的構(gòu)造函數(shù)的。函數(shù)對(duì)象 參數(shù)只能使用那些到定義Lambda為止時(shí)Lambda所在作用范圍內(nèi)可見(jiàn)的局部變量(包括Lambda所在類的this)。函數(shù)對(duì)象參數(shù)有以下形式:
1、空。沒(méi)有使用任何函數(shù)對(duì)象參數(shù)。
2、=。函數(shù)體內(nèi)可以使用Lambda所在作用范圍內(nèi)所有可見(jiàn)的局部變量(包括Lambda所在類的this),并且是值傳遞方式(相當(dāng)于編譯器自動(dòng)為我們按值傳遞了所有局部變量)。
3、&。函數(shù)體內(nèi)可以使用Lambda所在作用范圍內(nèi)所有可見(jiàn)的局部變量(包括Lambda所在類的this),并且是引用傳遞方式(相當(dāng)于編譯器自動(dòng)為我們按引用傳遞了所有局部變量)。
4、this。函數(shù)體內(nèi)可以使用Lambda所在類中的成員變量。
5、a。將a按值進(jìn)行傳遞。按值進(jìn)行傳遞時(shí),函數(shù)體內(nèi)不能修改傳遞進(jìn)來(lái)的a的拷貝,因?yàn)槟J(rèn)情況下函數(shù)是const的。要修改傳遞進(jìn)來(lái)的a的拷貝,可以添加mutable修飾符。
6、&a。將a按引用進(jìn)行傳遞。
7、a, &b。將a按值進(jìn)行傳遞,b按引用進(jìn)行傳遞。
8、=,&a, &b。除a和b按引用進(jìn)行傳遞外,其他參數(shù)都按值進(jìn)行傳遞。
9、&, a, b。除a和b按值進(jìn)行傳遞外,其他參數(shù)都按引用進(jìn)行傳遞。
二、(操作符重載函數(shù)參數(shù)),標(biāo)識(shí)重載的()操作符的參數(shù),沒(méi)有參數(shù)時(shí),這部分可以省略。參數(shù)可以通過(guò)按值(如:(a,b))和按引用(如:(&a,&b))兩種方式進(jìn)行傳遞。
三、mutable或exception聲明,這部分可以省略。按值傳遞函數(shù)對(duì)象參數(shù)時(shí),加上mutable修飾符后,可以修改按值傳遞進(jìn)來(lái)的拷貝(注意 是能修改拷貝,而不是值本身)。exception聲明用于指定函數(shù)拋出的異常,如拋出整數(shù)類型的異常,可以使用throw(int)。
四、->返回值類型,標(biāo)識(shí)函數(shù)返回值的類型,當(dāng)返回值為void,或者函數(shù)體中只有一處return的地方(此時(shí)編譯器可以自動(dòng)推斷出返回值類型)時(shí),這部分可以省略。
五、{函數(shù)體},標(biāo)識(shí)函數(shù)的實(shí)現(xiàn),這部分不能省略,但函數(shù)體可以為空。
下面給出了一段示例代碼,用于演示上述提到的各種情況,代碼中有簡(jiǎn)單的注釋可作為參考。

class CTest
{
public:
CTest() : m_nData(20) { NULL; }
void TestLambda()
{
vector<int> vctTemp;
vctTemp.push_back(1);
vctTemp.push_back(2);
// 無(wú)函數(shù)對(duì)象參數(shù),輸出:1 2
{
for_each(vctTemp.begin(), vctTemp.end(), [](int v){ cout << v << endl; });
}
// 以值方式傳遞作用域內(nèi)所有可見(jiàn)的局部變量(包括this),輸出:11 12
{
int a = 10;
for_each(vctTemp.begin(), vctTemp.end(), [=](int v){ cout << v+a << endl; });
}
// 以引用方式傳遞作用域內(nèi)所有可見(jiàn)的局部變量(包括this),輸出:11 13 12
{
int a = 10;
for_each(vctTemp.begin(), vctTemp.end(), [&](int v)mutable{ cout << v+a << endl; a++; });
cout << a << endl;
}
// 以值方式傳遞局部變量a,輸出:11 13 10
{
int a = 10;
for_each(vctTemp.begin(), vctTemp.end(), [a](int v)mutable{ cout << v+a << endl; a++; });
cout << a << endl;
}
// 以引用方式傳遞局部變量a,輸出:11 13 12
{
int a = 10;
for_each(vctTemp.begin(), vctTemp.end(), [&a](int v){ cout << v+a << endl; a++; });
cout << a << endl;
}
// 傳遞this,輸出:21 22
{
for_each(vctTemp.begin(), vctTemp.end(), [this](int v){ cout << v+m_nData << endl; });
}
// 除b按引用傳遞外,其他均按值傳遞,輸出:11 12 17
{
int a = 10;
int b = 15;
for_each(vctTemp.begin(), vctTemp.end(), [=, &b](int v){ cout << v+a << endl; b++; });
cout << b << endl;
}
// 操作符重載函數(shù)參數(shù)按引用傳遞,輸出:2 3
{
for_each(vctTemp.begin(), vctTemp.end(), [](int &v){ v++; });
for_each(vctTemp.begin(), vctTemp.end(), [](int v){ cout << v << endl; });
}
// 空的Lambda表達(dá)式
{
[](){}();
[]{}();
}
}
private:
int m_nData;
};
