程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> C語言 >> C++ >> 關於C++ >> C++ 中lambda表達式的編譯器實現原理

C++ 中lambda表達式的編譯器實現原理

編輯:關於C++

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表達式的編譯器實現原理,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對網站的支持!

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved