溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

C++中l(wèi)ambda表達(dá)式的編譯器怎么用

發(fā)布時間:2021-07-15 14:39:31 來源:億速云 閱讀:158 作者:小新 欄目:編程語言

這篇文章主要介紹C++中l(wèi)ambda表達(dá)式的編譯器怎么用,文中介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們一定要看完!

什么是Lambda?

C++ 11加入了一個非常重要的特性——Lambda表達(dá)式。營里(戴維營)的兄弟都對Objective-C很熟悉,許多人多block情有獨鐘,將各種回調(diào)函數(shù)、代理通通都用它來實現(xiàn)。甚至有人選擇用FBKVOController、BlocksKit等開源框架將KVO、控件事件處理都改為通過block解決。原因就是簡單、方便、直觀,函數(shù)的定義和使用出現(xiàn)在同一個地方。這里的Lambda表達(dá)式實際上和block非常類似,當(dāng)然如果你用它和Swift語言的閉包比較,那就是一回事了。

現(xiàn)在,Android已經(jīng)全面轉(zhuǎn)向C++11/14標(biāo)準(zhǔn)了,看代碼的話,很多地方變化很大,新標(biāo)準(zhǔn)真的是有點顛覆性的,感覺已經(jīng)不會C++了。今天有看到lambda表達(dá)式,突然想看一下,這貨是怎么實現(xiàn)的,如下,寫了個例子,分別調(diào)用3個lambda表達(dá)式:

#include <stdlib.h> 
#include <stdio.h> 
//1. 無參數(shù) 
auto hello = [] () {printf( "Hello world!\n");}; 
// 2. 一個參數(shù) 
auto hello_int = [] (int val){ printf("the value is %d\n", val); }; 
int main(int argc, char **argv) { 
 hello(); 
 hello_int(argc); 
 // 3. 帶捕獲列表的lambda表達(dá)式 
 auto lambda = [argc, argv]() {printf("param: %d, path is:%s\n", argc, argv[0]);}; 
 lambda(); 
 return 0; 
}

很簡單定義三個lambda表達(dá)式,lambda表達(dá)式就不細(xì)說是什么了,基本上是介紹新標(biāo)準(zhǔn)的書,都會說的很明白。這里想看一下,具體編譯器是怎么實現(xiàn)表達(dá)式的呢?第一印象,應(yīng)該是表達(dá)式按照內(nèi)聯(lián)函數(shù)的方式實現(xiàn)的吧,調(diào)用的地方自動展開,這樣參數(shù)、捕獲列表啥的都很好實現(xiàn)。

簡單看一下,編譯時不優(yōu)化,反匯編看一下,如下,調(diào)用的main函數(shù):

上面按順序調(diào)用的,就是代碼中對應(yīng)的三個lambda表達(dá)式。從這個反匯編看,貌似和猜想的不一樣是調(diào)了函數(shù)不是內(nèi)聯(lián)展開。
紅色框住的調(diào)用函數(shù)[藍(lán)色是實際的符號,灰色是demangle后的,分析看這個],分別是:

$_0::operator()(void)  
$_1::operator()(int) 
main::$_2::operator() const(void)

這是三個重載的()操作符.. 調(diào)用前看到有壓入this參數(shù),這是對象的方法調(diào)用,從反匯編看,是棧上創(chuàng)建對象,然后直接使用,使用是通過operator()..

這貨不就是函數(shù)對象么????

OK,那就明白了,lambda表達(dá)式,編譯器自動轉(zhuǎn)換成函數(shù)對象執(zhí)行。。。。

上面的例子,編譯器轉(zhuǎn)換的如下:

#include <stdlib.h> 
#include <stdio.h> 
class $_0 { 
public: 
 void operator() { 
 printf( "Hello world!\n"); 
 } 
}; 
class $_1 { 
public: 
 void operator(int va) { 
  printf("the value is %d\n", val); 
 } 
}; 
class main::$_2 { 
public: 
 main::$_2(int i, char **v): argc(i), argv(v) {} 
 //帶捕獲列表的,不能修改捕獲列表... 
 void operator() const { 
 printf("param: %d, path is:%s\n", argc, argv[0]); 
 } 
private: 
 int argc; 
 char **argv; 
} 
int main(int argc, char **argv) { 
 $_0 hello; 
 hello(); 
 $_1 hello_int; 
 hello_int(argc); 
 main::$_2 lambda(argc, argv); 
 lambda(); 
 return 0; 
}

這里,lambda轉(zhuǎn)換后的符號,是編譯器自動生成的,看起來稍有點別扭。

以上是“C++中l(wèi)ambda表達(dá)式的編譯器怎么用”這篇文章的所有內(nèi)容,感謝各位的閱讀!希望分享的內(nèi)容對大家有幫助,更多相關(guān)知識,歡迎關(guān)注億速云行業(yè)資訊頻道!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

c++
AI