【原】C++ 11 Lambda運算式

來源:互聯網
上載者:User

       C++ 11中的Lambda運算式用於定義並建立匿名的函數對象,以簡化編程工作。Lambda的文法形式如下:
              [函數對象參數] (操作符重載函數參數) mutable或exception聲明 ->傳回值類型 {函數體}
      可以看到,Lambda主要分為五個部分:[函數對象參數]、(操作符重載函數參數)、mutable或exception聲明、->傳回值類型、{函數體}。下面分別進行介紹。
      一、[函數對象參數],標識一個Lambda的開始,這部分必須存在,不能省略。函數對象參數是傳遞給編譯器自動產生的函數對象類的建構函式的。函數對象參數只能使用那些到定義Lambda為止時Lambda所在作用範圍內可見的局部變數(包括Lambda所在類的this)。函數對象參數有以下形式:
           1、空。沒有使用任何函數對象參數。
           2、=。函數體內可以使用Lambda所在作用範圍內所有可見的局部變數(包括Lambda所在類的this),並且是值傳遞方式(相當於編譯器自動為我們按值傳遞了所有局部變數)。
           3、&。函數體內可以使用Lambda所在作用範圍內所有可見的局部變數(包括Lambda所在類的this),並且是引用傳遞方式(相當於編譯器自動為我們按引用傳遞了所有局部變數)。
           4、this。函數體內可以使用Lambda所在類中的成員變數。
           5、a。將a按值進行傳遞。按值進行傳遞時,函數體內不能修改傳遞進來的a的拷貝,因為預設情況下函數是const的。要修改傳遞進來的a的拷貝,可以添加mutable修飾符。
           6、&a。將a按引用進行傳遞。
           7、a, &b。將a按值進行傳遞,b按引用進行傳遞。
           8、=,&a, &b。除a和b按引用進行傳遞外,其他參數都按值進行傳遞。
           9、&, a, b。除a和b按值進行傳遞外,其他參數都按引用進行傳遞。
      二、(操作符重載函數參數),標識重載的()操作符的參數,沒有參數時,這部分可以省略。參數可以通過按值(如:(a,b))和按引用(如:(&a,&b))兩種方式進行傳遞。
      三、mutable或exception聲明,這部分可以省略。按值傳遞函數對象參數時,加上mutable修飾符後,可以修改按值傳遞進來的拷貝(注意是能修改拷貝,而不是值本身)。exception聲明用於指定函數拋出的異常,如拋出整數類型的異常,可以使用throw(int)。
      四、->傳回值類型,標識函數傳回值的類型,當傳回值為void,或者函數體中只有一處return的地方(此時編譯器可以自動推斷出傳回值類型)時,這部分可以省略。
      五、{函數體},標識函數的實現,這部分不能省略,但函數體可以為空白。
      下面給出了一段範例程式碼,用於示範上述提到的各種情況,代碼中有簡單的注釋可作為參考。

class CTest
{
public:
 CTest() : m_nData(20) { NULL; }
 void TestLambda()
 {
  vector<int> vctTemp;
  vctTemp.push_back(1);
  vctTemp.push_back(2);

  // 無函數對象參數,輸出:1 2
  {
   for_each(vctTemp.begin(), vctTemp.end(), [](int v){ cout << v << endl; });
  }

  // 以值方式傳遞範圍內所有可見的局部變數(包括this),輸出:11 12
  {
   int a = 10;
   for_each(vctTemp.begin(), vctTemp.end(), [=](int v){ cout << v+a << endl; });
  }

  // 以引用方式傳遞範圍內所有可見的局部變數(包括this),輸出:11 13 12
  {
   int a = 10;
   for_each(vctTemp.begin(), vctTemp.end(), [&](int v)mutable{ cout << v+a << endl; a++; });
   cout << a << endl;
  }

  // 以值方式傳遞局部變數a,輸出:11 13 10
  {
   int a = 10;
   for_each(vctTemp.begin(), vctTemp.end(), [a](int v)mutable{ cout << v+a << endl; a++; });
   cout << a << endl;
  }

  // 以引用方式傳遞局部變數a,輸出:11 13 12
  {
   int a = 10;
   for_each(vctTemp.begin(), vctTemp.end(), [&a](int v){ cout << v+a << endl; a++; });
   cout << a << endl;
  }

  // 傳遞this,輸出:21 22
  {
   for_each(vctTemp.begin(), vctTemp.end(), [this](int v){ cout << v+m_nData << endl; });
  }

  // 除b按引用傳遞外,其他均按值傳遞,輸出:11 12 17
  {
   int a = 10;
   int b = 15;
   for_each(vctTemp.begin(), vctTemp.end(), [=, &b](int v){ cout << v+a << endl; b++; });
   cout << b << endl;
  }

  // 操作符重載函數參數按引用傳遞,輸出:2 3
  {
   for_each(vctTemp.begin(), vctTemp.end(), [](int &v){ v++; });
   for_each(vctTemp.begin(), vctTemp.end(), [](int v){ cout << v << endl; });
  }

  // 空的Lambda運算式
  {
   [](){}();
   []{}();
  }
 }

private:
 int m_nData;
};

 

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.