C++ 中l(wèi)ambda表達(dá)式的編譯器實(shí)現(xiàn)原理
什么是Lambda?
C++ 11加入了一個非常重要的特性——Lambda表達(dá)式。營里(戴維營)的兄弟都對Objective-C很熟悉,許多人多block情有獨(dú)鐘,將各種回調(diào)函數(shù)、代理通通都用它來實(shí)現(xiàn)。甚至有人選擇用FBKVOController、BlocksKit等開源框架將KVO、控件事件處理都改為通過block解決。原因就是簡單、方便、直觀,函數(shù)的定義和使用出現(xiàn)在同一個地方。這里的Lambda表達(dá)式實(shí)際上和block非常類似,當(dāng)然如果你用它和Swift語言的閉包比較,那就是一回事了。
現(xiàn)在,Android已經(jīng)全面轉(zhuǎn)向C++11/14標(biāo)準(zhǔn)了,看代碼的話,很多地方變化很大,新標(biāo)準(zhǔn)真的是有點(diǎn)顛覆性的,感覺已經(jīng)不會C++了。今天有看到lambda表達(dá)式,突然想看一下,這貨是怎么實(shí)現(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)的書,都會說的很明白。這里想看一下,具體編譯器是怎么實(shí)現(xiàn)表達(dá)式的呢?第一印象,應(yīng)該是表達(dá)式按照內(nèi)聯(lián)函數(shù)的方式實(shí)現(xiàn)的吧,調(diào)用的地方自動展開,這樣參數(shù)、捕獲列表啥的都很好實(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)色是實(shí)際的符號,灰色是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)換后的符號,是編譯器自動生成的,看起來稍有點(diǎn)別扭。
以上所述是小編給大家介紹的C++ 中l(wèi)ambda表達(dá)式的編譯器實(shí)現(xiàn)原理,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復(fù)大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!
相關(guān)文章
C++?opencv圖像處理實(shí)現(xiàn)圖片邊緣檢測示例
這篇文章主要為大家介紹了C++?opencv實(shí)現(xiàn)圖片邊緣檢測示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-05-05
Cocos2d-x保存用戶游戲數(shù)據(jù)CCUserDefault類使用實(shí)例
這篇文章主要介紹了Cocos2d-x保存用戶游戲數(shù)據(jù)CCUserDefault類使用實(shí)例,需要的朋友可以參考下2014-09-09
C++數(shù)據(jù)結(jié)構(gòu)與算法之判斷一個鏈表是否為回文結(jié)構(gòu)的方法
這篇文章主要介紹了C++數(shù)據(jù)結(jié)構(gòu)與算法之判斷一個鏈表是否為回文結(jié)構(gòu)的方法,結(jié)合實(shí)例形式分析了回文結(jié)構(gòu)并結(jié)合實(shí)例給出了C++判斷回文的操作技巧,需要的朋友可以參考下2017-05-05
C++ OpenCV實(shí)戰(zhàn)之網(wǎng)孔檢測的實(shí)現(xiàn)
這篇文章主要介紹了如何利用C++和OpenCV實(shí)現(xiàn)網(wǎng)孔檢測,文中的示例代碼講解詳細(xì),對我們學(xué)習(xí)OpenCV有一定幫助,感興趣的小伙伴可以了解一下2022-05-05
Matlab中圖像數(shù)字水印算法的原理與實(shí)現(xiàn)詳解
數(shù)字水印技術(shù)作為信息隱藏技術(shù)的一個重要分支,是將信息(水印)隱藏于數(shù)字圖像、視頻、音頻及文本文檔等數(shù)字媒體中,從而實(shí)現(xiàn)隱秘傳輸、存儲、標(biāo)注、身份識別、版權(quán)保護(hù)和防篡改等目的。本文就來講講圖像數(shù)字水印算法的原理與實(shí)現(xiàn),感興趣的可以了解一下2023-04-04
C++實(shí)現(xiàn)strcmp字符串比較的深入探討
本篇文章是對使用C++實(shí)現(xiàn)strcmp字符串比較進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05
C/C++?Qt?運(yùn)用JSON解析庫的實(shí)例代碼
這篇文章主要介紹了C/C++?Qt?運(yùn)用JSON解析庫的相關(guān)知識,通過代碼依次解析這個json文件中的每一個參數(shù),代碼簡單易懂,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-01-01

