溫馨提示×

溫馨提示×

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

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

C++11新引入的lambda表達式怎么用

發(fā)布時間:2021-09-16 09:49:57 來源:億速云 閱讀:138 作者:柒染 欄目:編程語言

本篇文章為大家展示了C++11新引入的lambda表達式怎么用,內(nèi)容簡明扼要并且容易理解,絕對能使你眼前一亮,通過這篇文章的詳細(xì)介紹希望你能有所收獲。

ISO C++ 11 標(biāo)準(zhǔn)的一大亮點是引入Lambda表達式。基本語法如下:

[capture list] (parameter list) ->return type { function body }

簡單的講一下各個部分的作用

1.[capture list]捕獲列表,捕獲到函數(shù)體中,使得函數(shù)體可以訪問
2.(parameter list)參數(shù)列表,用來表示lambda表達式的參數(shù)列表
3.->return type函數(shù)返回值 {function body}就是函數(shù)體

lambda表達式可以理解為一個匿名函數(shù)(但本質(zhì)并不是),如果要使用lambda表達式聲明的函數(shù),需要給他“命名”

lambda表達式可以表示閉包,因為本身就是這個類
閉包是指可以包含自由變量的代碼塊 (未綁定到特定的對象:舉個栗子std::function就可以產(chǎn)生一個對象,或者一個未指向任何函數(shù)的函數(shù)指針)

閉包講的通俗一些有以下幾點

1.自帶上下文的函數(shù),閉包可以儲存運行時需要的上下文,這樣就可以在上下文不存在的時候還可以使用閉包(變量a生命周期到了被銷毀,但是在閉包中還可以拿來用)
2.可以把閉包看成一個重載了operator()的類,帶有狀態(tài)的意思就可以解釋成通過this指針使用成員變量
3.capture list就是lambda表達式實現(xiàn)閉包的方式

簡單使用的例子

--------------------------------------------------------------------------------

C++11為auto提供了新的功能,如他的名字一般,現(xiàn)在可以看成自動適應(yīng)類型,可以適應(yīng)多數(shù)類型
使用auto來代替變量的類型,前提是被明確類型的初始化變量初始化的,可以使用auto關(guān)鍵字

 auto f = [](){}; 
 auto f = [](int a, int b)->int {return a + b; };
 f(1, 2);//需要這么使用

只要是函數(shù)類型就都可以使用這個lambda表達式

 typedef int(*FUNC)(int a, int b);
 int main()
 {
   FUNC a= [](int a, int b) {return a + b; };
  
   printf("%d\n", a(1, 2));
 }

聲明函數(shù)的方法都可以接收不帶捕獲列表的lambda表達式

 typedef std::function<int(int a, int b)> FUNC;
 int main()
 {
   FUNC a= [](int a, int b) {return a + b; };
  
   printf("%d\n", a(1, 2));
 }

--------------------------------------------------------------------------------

lambda表達式中capture list的用法

 int func(int a, int b, std::function<int(int, int)> f)
 {
   return f(a, b);
 }
 
 
 int a=1;
 int b=2;
 int c=3;
 int d = func(a, b, [a, &b](int m, int n) {
 
     printf("a = %d\n", a); // a是通過值傳遞捕獲,mutable只在函數(shù)體內(nèi)修改有效
     printf("b = %d\n", b); // b是引用傳遞捕獲,mutable可以對外部b造成影響
 
                //printf("c = %d\n", c); // c不可訪問
 
     return m + n;
   });
 typedef int(*FUNC)(int m, int n,std::function<int(int ,int )> f);
 
 void test()
 {
   FUNC oho;
   int a = 10;
   int b = 20;
   auto func = [&a, &b](int m, int n) {printf("a:%d b:%d\n", a, b); return m + n; };
   
 }

1.[]空。沒有使用任何函數(shù)對象參數(shù)。
2.[=]。函數(shù)體內(nèi)可以使用Lambda所在作用范圍內(nèi)所有可見的局部變量(包括Lambda所在類的this),并且是值傳遞方式(相當(dāng)于編譯器自動為我們按值傳遞了所有局部變量)。
3.[&]。函數(shù)體內(nèi)可以使用Lambda所在作用范圍內(nèi)所有可見的局部變量(包括Lambda所在類的this),并且是引用傳遞方式(相當(dāng)于編譯器自動為我們按引用傳遞了所有局部變量)。
4.[this]。函數(shù)體內(nèi)可以使用Lambda所在類中的成員變量。
5.[a]。將a按值進行傳遞。按值進行傳遞時,函數(shù)體內(nèi)不能修改傳遞進來的a的拷貝,因為默認(rèn)情況下函數(shù)是const的。要修改傳遞進來的a的拷貝,可以添加mutable修飾符。
6.[&a]。將a按引用進行傳遞。
7.[a, &b]。將a按值進行傳遞,b按引用進行傳遞。
8.[=,&a, &b]。除a和b按引用進行傳遞外,其他參數(shù)都按值進行傳遞。注意=符號的位置必須在頭一個
9.[&, a, b]。除a和b按值進行傳遞外,其他參數(shù)都按引用進行傳遞。&符號的位置必須在頭一個

當(dāng)你想改變通過傳值方式捕捉的變量的時候就要添加mutable

[a, &b, &b2](int m, int n)mutable {a *= 2; return m*n; }:

--------------------------------------------------------------------------------

lambda表達式的其他用法

 class A
 {
 public:
   A();
   ~A();
   void test()
   {
     auto f = [this](int m, int n) {printf("%d\n", a); };
   }
     
 private:
   int a;
 };

lambda表達式本質(zhì)是一種閉包類型,雖然他可以賦值給函數(shù)指針,但是只限于在捕獲列表為空的時候,當(dāng)捕獲列表有值的時候,應(yīng)該使用auto來接收lambda表達式,或者用std::function也是可以的

 main::__l2::<lambda_eb7b0a89c14bee3d2620c108ffb635c6>
 //這是一個lambda表達式在VS2015環(huán)境下顯示的類型,不用auto用什么來接收調(diào)用他呢?

本質(zhì)來說lambda表達式之間是不允許賦值的

 auto a = [](int m, int n) {return m + n; };
 auto b = [](int m, int n) {return m - n; };
 a = b;

操作非法,因為閉包類型不允許使用賦值操作符,但是函數(shù)指針可以,也就是可以有下面的操作

 typedef int(*FUNC)(int a, int b);
 int main()
 {
   FUNC a = [](int a, int b) {return a + b; };
   FUNC b = [](int a, int b) {return a + b; };
   a = b;
   return 0;
 }

std::function之間也是可以賦值的,他就可以辦到有capture list的lambda表達式進行賦值

 typedef std::function<int(int,int)> FUNC;
 int m = 10;
 int n = 20;
 FUNC a = [m, n](int a, int b){printf("%d\n", m); return a + b; };
 FUNC b = [m, n](int a, int b){return a + b; };
 b = a;
 b(1, 2);
 //執(zhí)行結(jié)果是可以把m打印出來的

上述內(nèi)容就是C++11新引入的lambda表達式怎么用,你們學(xué)到知識或技能了嗎?如果還想學(xué)到更多技能或者豐富自己的知識儲備,歡迎關(guān)注億速云行業(yè)資訊頻道。

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

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

AI