什么是lambda表達(dá)式
lambda表達(dá)式是一個(gè)可調(diào)用的代碼單元橡疼,我們可以理解為一個(gè)未命名的內(nèi)聯(lián)函數(shù),當(dāng)定義一個(gè)lambda時(shí),編譯器會(huì)生成一個(gè)與lambda對(duì)應(yīng)的類類型犀斋。
從lambda生成的類包含它所捕獲的變量的數(shù)據(jù)成員律胀,被捕獲的變量在lambda創(chuàng)建時(shí)進(jìn)行拷貝或者引用(而不是調(diào)用時(shí))宋光。
lambda表達(dá)式格式
一個(gè)lambda表達(dá)式的聲明格式如下:
[capture list] (params list) mutable exception-> return type { function body }
各項(xiàng)具體含義:
- capture list:捕獲外部變量列表
- params list:形參列表
- mutable:用來說用是否可以修改捕獲的變量
- exception:異常設(shè)定
- return type:返回類型
- function body:函數(shù)體
mutable和exception不是必須的,可以省略
[capture list] (params list)-> return type { function body }
auto lambdaFunc = []() ->const char* {return "hello world"; };
如果省略返回類型炭菌,lambda根據(jù)函數(shù)體中的代碼推斷返回類型罪佳,如果函數(shù)體有return語句,則返回類型從返回的表達(dá)式的類型推斷而來黑低,否則返回類型為void赘艳。
[capture list] (params list){ function body }
auto lambdaFunc = [] (){return "hello world"; };
參數(shù)列表為空且省略了返回類型,mutable,exception時(shí)蕾管,可以省略()
[capture list] -> return type { function body }
auto lambdaFunc = []{return "hello world"; };
使用捕獲列表
lambda通過捕獲列表指出它要使用哪些外部變量枷踏,然后就可以再函數(shù)體中訪問這些變量了。
int main()
{
const char* helloStr = "hello world";
auto lambdaFunc = [helloStr]{return helloStr; };
std::cout << lambdaFunc() << std::endl;
system("pause");
return 0;
}
值捕獲與引用捕獲
類似參數(shù)傳遞掰曾,變量的捕捉方式也可以是值或者引用旭蠕,對(duì)于值捕獲的變量,lambda在創(chuàng)建時(shí)會(huì)對(duì)其進(jìn)行拷貝婴梧,而引用捕獲則不同下梢,我們?cè)趌ambda內(nèi)使用此變量時(shí),實(shí)際上使用的是引用所綁定的變量塞蹭,不會(huì)發(fā)生拷貝孽江。
值捕獲,輸出hello world
int main()
{
std::string helloStr = "hello world";
auto lambdaFunc = [helloStr]{return helloStr; };
helloStr = "hello xy";
std::cout << lambdaFunc() << std::endl;
system("pause");
return 0;
}
引用捕獲番电,輸出hello xy
int main()
{
std::string helloStr = "hello world";
auto lambdaFunc = [&helloStr]{return helloStr; };
helloStr = "hello xy";
std::cout << lambdaFunc() << std::endl;
system("pause");
return 0;
}
隱式捕獲
除了指定捕獲變量外岗屏,還可以讓編譯器根據(jù)函數(shù)體中的代碼來推斷需要捕獲哪些變量,隱式捕獲有兩種漱办,[=]和[&]这刷。[=]表示以值捕獲的方式捕獲外部變量,[&]表示以引用捕獲的方式捕獲外部變量娩井。
int main()
{
int value1 = 10;
int value2 = 100;
auto lambdaFunc1 = [&]{return ++value1; };
auto lambdaFunc2 = [=] {return value2; };
std::cout << lambdaFunc1() << std::endl;
std::cout << lambdaFunc2() << std::endl;
system("pause");
return 0;
}
混合捕獲
lambda還支持混合方式捕獲變量:
- [=, &x]暇屋,變量x以引用形式捕獲,其余變量以傳值形式捕獲洞辣。
- [&, x]咐刨,變量x以值的形式捕獲,其余變量以引用形式捕獲扬霜。
int main()
{
int value1 = 10;
int value2 = 100;
auto lambdaFunc = [&,value2]{return ++value1+value2; };
std::cout << lambdaFunc() << std::endl;
system("pause");
return 0;
}
以值的形式捕獲this指針
class Person
{
public:
int height_=170;
void printHeight()
{
auto lambdaFunc = [this] {return this->height_;};
std::cout << lambdaFunc() << std::endl;
}
};
int main()
{
Person person;
person.printHeight();
system("pause");
return 0;
}
修改值捕獲的變量
如果以值傳遞方式捕捉外部變量定鸟,那么函數(shù)體中不能修改該變量(可以理解為加了一個(gè)const修飾符),否則會(huì)引起編譯錯(cuò)誤著瓶,如下代碼:
int main()
{
int value1 = 10;
auto lambdaFunc = [value1](){return ++value1; };
std::cout << lambdaFunc() << std::endl;
system("pause");
return 0;
}
這個(gè)時(shí)候使用mutable關(guān)鍵字即可修改捕捉的變量(取消const修飾)联予。
int main()
{
int value1 = 10;
auto lambdaFunc = [value1]()mutable{return ++value1; };
std::cout << lambdaFunc() << std::endl;
system("pause");
return 0;
}