<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
Lambda(匿名函數)表示式是C++11最重要的特性之一,lambda來源於函數語言程式設計的概念,也是現代程式語言的一個特點。
優點如下:
lambda表示式定義了一個匿名函數,並且可以捕獲一定範圍內的變數。語法形式如下:
[ capture ] ( params ) opt -> ret { body; };
一個完整的lambda表示式是這樣:
auto f = [](int a) -> int {return a + 1;}; cout << f(3) << endl; //輸出4
以上定義了一個完整的lambda,但是在實際的使用中,可以省略其返回值的定義,編譯器會根據return語句進行自動推導返回值型別。
省略過後如下:
auto f = [](int a) {return a + 1;};
需要注意的是,初始化列表不能用於返回值的自動推導:
如:auto f = [](){return {1,2};}; //error:無法推導返回值型別
另外,如果表示式沒有參數列時,也可以省略,如:
auto f = []{return 1;};
lambda表示式可以通過捕獲列表捕獲一定範圍內的變數,主要有以下幾種情況:
通過範例來看具體用法:
class A { public: int i_ = 0; void func(int x,int y) { auto x1 = []{return i_;}; // error,沒有捕獲外部變數 auto x2 = [=]{return i_ + x + y;}; //ok,按值捕獲所有外部變數 auto x3 = [&]{return i_ + x + y;}; //ok,按參照捕獲所有外部變數 auto x4 = [this]{return i_;}; //ok,捕獲this指標 auto x5 = [this]{return i_ + x + y;}; //error,沒有捕獲x和y變數 auto x6 = [this,x,y]{return i_ + x + y;}; //ok,捕獲了this指標和x、y變數 auto x7 = [this]{return i_++;}; //ok,捕獲了this指標,修改成員變數的值 } };
int a = 0 , b = 0 ; auto f1 = []{return a;}; // error,沒有捕獲外部變數 auto f2 = [&]{return a++;}; //ok,捕獲所有外部變數,並對a變數自加 auto f3 = [=]{return a;}; //ok,捕獲所有外部變數,並返回a auto f4 = [=]{return a++;}; //error,a變數是以複製方式捕獲的,不能修改 auto f5 = [a]{return a+b;}; //error,沒有捕獲b變數 auto f6 = [a,&b]{return a+ (b++);}; //ok,捕獲a以及b的參照,對b進行自加 auto f7 = [=,&b]{return a+ (b++);}; //ok, 捕獲所有外部變數和b的參照,對b進行自加
需要注意的是,lambda無法修改按值捕獲的外部變數,如果需要修改外部變數,可以通過參照方式捕獲。
關於lambda表示式的延遲呼叫很容易出錯,如下:
int a = 0; auto f = [=]{return a;}; a += 1; cout << f() << endl;
以上範例中,lambda按值捕獲了所有外部變數,在捕獲的時候 a的值就已經被複制到 f 中了,之後a被修改,但是f裡面儲存的a仍然是捕獲時的值,所以最終輸出的是 0.
如果希望lambda表示式在呼叫的時候能夠存取外部變數,需要使用參照方式捕獲。
所以簡單來說,按值捕獲,外部變數會被複制一份儲存在lambda表示式變數中。
如果是按值捕獲並且又想修改外部變數,可以顯示指明lambda表示式為mutable:
int a = 0; auto f1 = [=]{return a++;}; //error,修改按值捕獲的外部變數 auto f2 = [=]() mutable {return a++;}; //ok
被mutable修飾的lambda表示式就算沒有引數也要寫明參數列。
lambda表示式的型別在C++11中被稱為“閉包型別”,它是一個特殊的,匿名的非nunion的型別。
可以認為它是帶有一個operator()的類,即仿函數。
我們可以通過std::function和std::bind來儲存和操作lambda表示式:
std::function<int(int)> f1 = [](int a){return a;}; std::function<int(void)> f2 = std::bind([](int a){return a;},123);
另外,對於沒有捕獲任何變數的lambda表示式,還可以被轉換成一個普通的函數指標:
using func_t = int(*)(int); func_t f = [](int a){return a;}; f(123);
lambda可以說是就地定義仿函數閉包的“語法 糖”。它的捕獲列表捕獲住任何外部變數,最終都會變為閉包型別的成員變數。而一個成員變數的類的operator(),如果能直接被轉換為普通的函數指標,那麼lambda表示式本身的this指標就丟掉了。而沒有捕獲任何外部變數的lambda表示式則不存在這個問題。
需要注意的是,沒有捕獲變數的lambda表示式可以直接轉換為函數指標,而捕獲變數的lambda表示式則不能轉換為函數指標。如下:
typedef void(*Ptr)(int*); Ptr p = [](int *p){delete p;}; //ok Ptr p1 = [&](int *p){delete p;}; //error
前面說到的按值捕獲無法修改捕獲的外部變數,因為按照C++標準,lambda表示式的operator()預設是const的,一個const成員函數是無法修改成員變數的值,而mutable的作用,就是取消operator()的const限制。
通過範例來看一下lambda的使用,在C++11之前,如果要用for_each函數將陣列中的偶數數量列印出來,程式碼如下:
#include <vector> #include <algorithm> class Count { public: Count(int &val):num(val){} void operator()(int val){ if(!(val & 1)){ ++num; } } private: int # }; int main() { std::vector<int> v = {1,2,3,4,5,6,7}; int count = 0; for_each(v.begin(),v.end(),Count(count)); std::cout << count << endl; return 0; }
如果使用lambda表示式,就可以簡化一下,真正使用閉包概念來替換這裡的仿函數。
#include <vector> #include <algorithm> int main() { std::vector<int> v = {1,2,3,4,5,6,7}; int count = 0; for_each(v.begin(),v.end(),[&count](int val){ if(!(val & 1)){ ++count; } }); std::cout << count << endl; return 0; }
lambda表示式的價值在於,就地封裝短小的功能閉包,方便地表達出我們希望執行的具體操作,並讓上下文結合的更加緊,程式碼更加簡潔,更靈活,也提高了開發效率及可維護性。
參考:《深入應用C++11》
到此這篇關於C++11 lambda(匿名函數)表示式詳細介紹的文章就介紹到這了,更多相關C++11 lambda表示式內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45