C++ 中lambda表達式的編譯器實現原理。本站提示廣大學習愛好者:(C++ 中lambda表達式的編譯器實現原理)文章只能為提供參考,不一定能成為您想要的結果。以下是C++ 中lambda表達式的編譯器實現原理正文
什麼是Lambda?
C++ 11加入了一個非常重要的特性——Lambda表達式。營裡(戴維營)的兄弟都對Objective-C很熟悉,許多人多block情有獨鐘,將各種回調函數、代理通通都用它來實現。甚至有人選擇用FBKVOController、BlocksKit等開源框架將KVO、控件事件處理都改為通過block解決。原因就是簡單、方便、直觀,函數的定義和使用出現在同一個地方。這裡的Lambda表達式實際上和block非常類似,當然如果你用它和Swift語言的閉包比較,那就是一回事了。
現在,Android已經全面轉向C++11/14標准了,看代碼的話,很多地方變化很大,新標准真的是有點顛覆性的,感覺已經不會C++了。今天有看到lambda表達式,突然想看一下,這貨是怎麼實現的,如下,寫了個例子,分別調用3個lambda表達式:
#include <stdlib.h> #include <stdio.h> //1. 無參數 auto hello = [] () {printf( "Hello world!\n");}; // 2. 一個參數 auto hello_int = [] (int val){ printf("the value is %d\n", val); }; int main(int argc, char **argv) { hello(); hello_int(argc); // 3. 帶捕獲列表的lambda表達式 auto lambda = [argc, argv]() {printf("param: %d, path is:%s\n", argc, argv[0]);}; lambda(); return 0; }
很簡單定義三個lambda表達式,lambda表達式就不細說是什麼了,基本上是介紹新標准的書,都會說的很明白。這裡想看一下,具體編譯器是怎麼實現表達式的呢?第一印象,應該是表達式按照內聯函數的方式實現的吧,調用的地方自動展開,這樣參數、捕獲列表啥的都很好實現。
簡單看一下,編譯時不優化,反匯編看一下,如下,調用的main函數:
上面按順序調用的,就是代碼中對應的三個lambda表達式。從這個反匯編看,貌似和猜想的不一樣是調了函數不是內聯展開。
紅色框住的調用函數[藍色是實際的符號,灰色是demangle後的,分析看這個],分別是:
$_0::operator()(void) $_1::operator()(int) main::$_2::operator() const(void)
這是三個重載的()操作符.. 調用前看到有壓入this參數,這是對象的方法調用,從反匯編看,是棧上創建對象,然後直接使用,使用是通過operator()..
這貨不就是函數對象麼????
OK,那就明白了,lambda表達式,編譯器自動轉換成函數對象執行。。。。
上面的例子,編譯器轉換的如下:
#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轉換後的符號,是編譯器自動生成的,看起來稍有點別扭。
以上所述是小編給大家介紹的C++ 中lambda表達式的編譯器實現原理,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對網站的支持!