C++ 中的Lambda表達式

来源:https://www.cnblogs.com/timefiles/archive/2022/07/27/CppLambda.html
-Advertisement-
Play Games

Lambda 表達式(lambda expression)是一個匿名函數,Lambda表達式基於數學中的λ演算得名,直接對應於其中的lambda抽象(lambda abstraction),是一個匿名函數,即沒有函數名的函數。Lambda表達式可以表示閉包(註意和數學傳統意義上的不同)。 閉包就是能... ...


目錄

簡介

Lambda 表達式(lambda expression)是一個匿名函數,Lambda表達式基於數學中的λ演算得名,直接對應於其中的lambda抽象(lambda abstraction),是一個匿名函數,即沒有函數名的函數。Lambda表達式可以表示閉包(註意和數學傳統意義上的不同)。
閉包就是能夠讀取其他函數內部變數的函數,可以理解成“定義在一個函數內部的函數“。在本質上,閉包是將函數內部和函數外部連接起來的橋梁。
C++中的Lambda表達式從C++11開始引入,完整的聲明如下:

[ 捕獲 ] <模板形參> 約束(可選)
( 形參 ) lambda說明符 約束(可選) { 函數體 }

上面的 <模板形參>約束(可選)lambda說明符 屬於較新的標準(c++17起),一般用的比較少,後面主要說明 [ 捕獲 ] 部分。
形參函數體 與具名函數的定義一致,沒有區別。

一個簡單的Lambda表達式應用場景,代碼如下:

#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;

int main()
{
    vector<int> vec{ 3, 4 };
    //降序排序
    sort(vec.begin(), vec.end(), [](int a, int b) {return a > b; });
    for (size_t i = 0; i < vec.size(); i++)
    {
        cout << vec[i] << endl;
    }
}

捕獲

捕獲是一個含有零或更多個捕獲符的逗號分隔列表,可以預設捕獲符開始。
預設捕獲符只有 &(以引用隱式捕獲被使用的自動變數)和=(以**複製隱式捕獲被使用的自動變數)。

  • 當預設捕獲符是 & 時,後繼的簡單捕獲符不能以 & 開始。
struct S2 { void f(int i); };
void S2::f(int i)
{
    [&]{};          // OK:預設以引用捕獲
    [&, i]{};       // OK:以引用捕獲,但 i 以值捕獲
    [&, &i] {};     // 錯誤:以引用捕獲為預設時的以引用捕獲
    [&, this] {};   // OK:等價於 [&]
    [&, this, i]{}; // OK:等價於 [&, i]
}
  • 當預設捕獲符是 = 時,後繼的簡單捕獲符必須以 & 開始,或者為 *this (C++17 起) 或 this (C++20 起)。
struct S2 { void f(int i); };
void S2::f(int i)
{
    [=]{};          // OK:預設以複製捕獲
    [=, &i]{};      // OK:以複製捕獲,但 i 以引用捕獲
    [=, *this]{};   // C++17 前:錯誤:無效語法
                    // C++17 起:OK:以複製捕獲外圍的 S2
    [=, this] {};   // C++20 前:錯誤:= 為預設時的 this
                    // C++20 起:OK:同 [=]
}
  • 任何捕獲符只可以出現一次,並且名字不能與形參相同:
struct S2 { void f(int i); };
void S2::f(int i)
{
    [i, i] {};        // 錯誤:i 重覆
    [this, *this] {}; // 錯誤:"this" 重覆(C++17)
 
    [i] (int i) {};   // 錯誤:形參和捕獲的名字相同
}

上面出現的兩個特殊的捕獲符作用如下:

  • this:當前對象的簡單的以引用捕獲
  • * this:當前對象的簡單的以複製捕獲

原理

先建一個簡單的Lambda表達式示例,代碼如下:

#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;

int main()
{
    int sum = 0;
    int std = 1;
    vector<int> vec{ 3, 4 };    
    for_each(vec.begin(), vec.end(), [&sum,std](int x) {sum += (x+std); });
    cout << sum << endl;
}

然後在C++ Insights中查看Lambda表達式展開後的代碼,完整代碼如下:

#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;

int main()
{
  int sum = 0;
  int std = 1;
  std::vector<int, std::allocator<int> > vec = std::vector<int, std::allocator<int> >{std::initializer_list<int>{3, 4}, std::allocator<int>()};
    
  class __lambda_11_38
  {
    public: 
    inline void operator()(int x) const
    {
      sum = sum + (x + std);
    }
    
    private: 
    int & sum;
    int std;
    public: 
    // inline /*constexpr */ __lambda_11_38(__lambda_11_38 &&) noexcept = default;
    __lambda_11_38(int & _sum, int & _std)
    : sum{_sum}
    , std{_std}
    {}
    
  };
  
  std::for_each(__gnu_cxx::__normal_iterator<int *, std::vector<int, std::allocator<int> > >(vec.begin()), __gnu_cxx::__normal_iterator<int *, std::vector<int, std::allocator<int> > >(vec.end()), __lambda_11_38(__lambda_11_38{sum, std}));
  std::cout.operator<<(sum).operator<<(std::endl);
  return 0;
}

可以看到Lambda表達式展開為類__lambda_11_38,捕獲的外部變數賦值到類的成員變數上,引用捕獲以指針賦值,複製捕獲直接拷貝。
__lambda_11_38重載了操作符(),它其實就是一個仿函數。

Lambda回調

在C++中可以使用模板、函數指針、抽象類和Lambda實現回調的效果,此處主要說明如何使用Lambdafunction在同步線程中實現回調的效果。
類模板 std::function 是通用多態函數包裝器,實例能存儲、複製及調用任何可複製構造 (CopyConstructible) 的可調用 (Callable) 目標——函數、 lambda 表達式、 bind 表達式或其他函數對象,還有指向成員函數指針和指向數據成員指針。
若 std::function 不含目標,則稱它為空,調用空 std::function 的目標導致拋出 std::bad_function_call 異常。

一個簡單的Lambda回調,類似於C#中的事件,代碼如下:

#include <iostream>
#include <vector>
#include <algorithm>
#include <functional>
using namespace std;

class Test
{
public:
    function<void(const int& num)> Func;
    void SetNum(int num) 
    {
        nowNum = num;
        OnFunc(nowNum);
    }
private:
    int nowNum;
    void OnFunc(const int& num)
    {
        if (Func)
        {
            // 在此處回調
            Func(num);		
        }
    }
};
int main()
{
    Test test;
    test.Func = [](const int& num)
    {
        cout << num << endl;
    };
    test.SetNum(100);
}

參考


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 前段時間在用Promise.all執行一個非常大批量的操作時遇到一個奇怪的問題。 這個Promise.all需要遍歷一個指定目錄中的所有文件,並以非同步的方式讀取文件內容併進行後續操作。由於目錄中的文件數目比較多(大約8000+),Promise.all在執行的過程中有許多文件讀取失敗,但是如果指定一 ...
  • 1 多人語音聊天功能介紹 本文展示瞭如何使用 ZEGO Express SDK 構造多人音視頻通話場景,即實現多對多實時音視頻聊天互動。用戶可在房間內與其餘用戶進行實時音視頻通話,互相推拉流。該場景可用於多人實時音視頻聊天、多人視頻會議等。 2 Web端實現多人語音聊天準備工作 在應用多人音視頻通話 ...
  • 這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 微信支付介紹 微信公眾號程式或微信小程式項目中基本都要有微信支付。支付場景基本就是點擊支付,然後手機底部拉起輸入微信支付密碼彈窗,密碼輸入正確後,支付完成。但是無論是微信小程式或微信公眾號程式都是在後端調用微信api介面進行統一下單,將接 ...
  • 需求: el-form 每行顯示兩列,底部按鈕居中 問題: 以前的解決辦法是: el-col, el-row。但是這裡只有一個 el-form-item 的 label 數據是已知的,其餘項都是迴圈得到的,數量不固定,因此不能採用 el-col 方式。 嘗試 嘗試過 item 左浮動,flex,底部 ...
  • 需求 1 :設置初始高亮 子組件: 父組件 需求 2 :高亮行的變化,需要把數據傳遞到兄弟組件中 解決辦法:EventBus 參考鏈接: http://t.csdn.cn/iwOJc main.js 中: 子組件 1 : 子組件 2 : 需求 3 :子組件把高亮行的 index 和表格總條數傳遞給父 ...
  • 從ES6開始增加了Promise類型,稱為了主導性的非同步編程機制。 期約Promise是一個有狀態的對象,可能處於如下三種狀態之一: 待定(pending) 兌現(fulfilled,或被稱為“解決”,resolved) 拒絕(rejected) pending是期約的最初始狀態。在這個狀態下,pr ...
  • 互聯網協議 1.C/S B/S 架構 client 基於網路通信 server browser 基於網路通信 server server端必須滿足的條件: 1. 穩定運行(網路. 硬體. 操作系統. 服務端軟體) 2. 服務端必須綁定一個固定的地址 2.什麼是互聯網 兩大要素: 1. 底層的物理鏈接 ...
  • 面向對象編程(中級) 筆記目錄:(https://www.cnblogs.com/wenjie2000/p/16378441.html) lntelliJ IDEA ●IDEA介紹(內容僅需瞭解) IDEA全稱Intelli IDEA 在業界被公認為最好的Java開發工具 IDEA是JetBrain ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...