函數模板_構造函數棧溢出

来源:https://www.cnblogs.com/chendasxian/archive/2023/09/25/17728872.html
-Advertisement-
Play Games

前言 最近寫一個任務隊列,可以支持存入返回值為void的任意函數對象。需要定義一個Task模板,來存儲函數對象以及參數。大致的實現如下: class Task { public: template <typename Func, typename... Args> Task(Func&& f, Ar ...


前言

最近寫一個任務隊列,可以支持存入返回值為void的任意函數對象。需要定義一個Task模板,來存儲函數對象以及參數。大致的實現如下:

class Task
{
public:
    template <typename Func, typename... Args>
    Task(Func&& f, Args &&...args)
        : func_(std::bind(std::forward<Func>(f), std::forward<Args>(args)...)) {}

    void operator()()
    {
        func_();
    }
private:
    std::function<void()> func_;
};

其中構造函數是一個函數模板,可以在編譯的時候,根據傳入的函數對象和參數,綁定生成std::function,存儲在func_中。

支持形如

auto f1 = [](int i, int j)
{
    std::cout << i << j;
};
auto f2 = [](int i, double j)
{
    std::cout << i << j;
};
Task t(f1, 5, 6);
Task t2(f2, 1, 2.3);

複製棧溢出

但下麵這個普通的“拷貝”,linux編譯正常,用clang編譯器的話會造成棧溢出。
auto t3 = t;
用vs看調用堆棧,發現一直在執行Task::<Task&>(Task & f)--->bind---- > binder等等,一直在執行構造函數。

我嘗試自定義拷貝構造,併在其中輸出。但發現拷貝函數根本沒有執行,而是在反覆執行函數模板。

Task(const Task & other) :func_(other.func_)
{
    std::cout << "copy ctor";
}

為什麼沒有調用拷貝構造

仔細研究發現調用堆棧,發現調用的是Task::<Task&>(Task& f),這並不是拷貝構造,這是函數模板自動生成的。

在https://cppinsights.io/這個網址,提供了編譯結果,可以看模板生成了那些函數。

#include <functional>
#include<iostream>

class Task
{

public:
    template<typename Func, typename ... Args>
    inline Task(Func&& f, Args &&... args)
        : func_{ std::bind(std::forward<Func>(f), std::forward<Args>(args)...) }
    {
    }


    /* First instantiated from: insights.cpp:37 */
#ifdef INSIGHTS_USE_TEMPLATE
    template<>
    inline Task<__lambda_29_13&, int, int>(__lambda_29_13& f, int&& __args1, int&& __args2)
        : func_{ std::function<void()>(std::bind(std::forward<__lambda_29_13&>(f), std::forward<int>(__args1), std::forward<int>(__args2))) }
    {
    }
#endif



    /* First instantiated from: insights.cpp:38 */
#ifdef INSIGHTS_USE_TEMPLATE
    template<>
    inline Task<__lambda_33_13&, int, double>(__lambda_33_13& f, int&& __args1, double&& __args2)
        : func_{ std::function<void()>(std::bind(std::forward<__lambda_33_13&>(f), std::forward<int>(__args1), std::forward<double>(__args2))) }
    {
    }
#endif



    /* First instantiated from: insights.cpp:39 */
#ifdef INSIGHTS_USE_TEMPLATE
    template<>
    inline Task<Task&>(Task& f)
        : func_{ std::function<void()>(std::bind(std::forward<Task&>(f))) }
    {
    }
#endif



#ifdef INSIGHTS_USE_TEMPLATE
    template<>
    inline Task<const Task&>(const Task& f);
#endif



    /* First instantiated from: functional:558 */
#ifdef INSIGHTS_USE_TEMPLATE
    template<>
    inline Task<Task>(Task&& f)
        : func_{ std::function<void()>(std::bind(std::forward<Task>(f))) }
    {
    }
#endif



#ifdef INSIGHTS_USE_TEMPLATE
    template<>
    inline Task<const std::_Bind<Task()>&>(const std::_Bind<Task()>& f);
#endif



#ifdef INSIGHTS_USE_TEMPLATE
    template<>
    inline Task<std::_Bind<Task()> >(std::_Bind<Task()>&& f);
#endif


    inline void operator()()
    {
        this->func_.operator()();
    }

    inline Task(const Task& other)
        : func_{ std::function<void()>(other.func_) }
    {
        std::operator<<(std::cout, "copy ctor");
    }


private:
    std::function<void()> func_;
public:
    // inline ~Task() noexcept = default;
};




int main()
{

    class __lambda_29_13
    {
    public:
        inline /*constexpr */ void operator()(int i, int j) const
        {
            std::cout.operator<<(i).operator<<(j);
        }

        using retType_29_13 = void (*)(int, int);
        inline constexpr operator retType_29_13 () const noexcept
        {
            return __invoke;
        };

    private:
        static inline /*constexpr */ void __invoke(int i, int j)
        {
            __lambda_29_13{}.operator()(i, j);
        }

    public:
        // inline /*constexpr */ __lambda_29_13 & operator=(const __lambda_29_13 &) /* noexcept */ = delete;
        // inline /*constexpr */ __lambda_29_13(const __lambda_29_13 &) noexcept = default;
        // inline /*constexpr */ __lambda_29_13(__lambda_29_13 &&) noexcept = default;

    };

    __lambda_29_13 f1 = __lambda_29_13{};

    class __lambda_33_13
    {
    public:
        inline /*constexpr */ void operator()(int i, double j) const
        {
            std::cout.operator<<(i).operator<<(j);
        }

        using retType_33_13 = void (*)(int, double);
        inline constexpr operator retType_33_13 () const noexcept
        {
            return __invoke;
        };

    private:
        static inline /*constexpr */ void __invoke(int i, double j)
        {
            __lambda_33_13{}.operator()(i, j);
        }

    public:
        // inline /*constexpr */ __lambda_33_13 & operator=(const __lambda_33_13 &) /* noexcept */ = delete;
        // inline /*constexpr */ __lambda_33_13(const __lambda_33_13 &) noexcept = default;
        // inline /*constexpr */ __lambda_33_13(__lambda_33_13 &&) noexcept = default;

    };

    __lambda_33_13 f2 = __lambda_33_13{};
    Task t = Task(f1, 5, 6);
    Task t2 = Task(f2, 1, 2.2999999999999998);
    Task t3 = Task(t);
    return 0;
}

可以看到最後t3是這麼生成的。
Task t3 = Task(t);
而Task類中有兩個形如這樣的函數;

//這個是拷貝構造函數

inline Task(const Task& other)
    : func_{ std::function<void()>(other.func_) }
{
    std::operator<<(std::cout, "copy ctor");
}

//這個是函數模板生成的一個構造函數,參數為Task &f。

#ifdef INSIGHTS_USE_TEMPLATE
template<>
inline Task<Task&>(Task& f)
    : func_{ std::function<void()>(std::bind(std::forward<Task&>(f))) }
{
}
#endif

這下明白為什麼沒有調用拷貝構造了,原來Task t3 = Task(t)中,等號右邊的Task(t)並不是拷貝構造函數。因為t是非常量左值,所以編譯器優先匹配模板函數的參數(Task & )。

為什麼會棧溢出

接下來是為什麼這個模板函數會遞歸構造,直至棧溢出。

觀察這個模板函數,發現其形參中有std::bind。而這個函數會複製拷貝傳入參數(這裡就是Task)。而複製Task並不會調用構造函數,而是調用這個函數模板,因此,一直遞歸調用直至棧溢出。

解決

既然非常量左值匹配不上拷貝構造,那就把返回值轉換成常量左值, 改成下麵這種形式。就沒問題了

auto t3 = static_cast<Task&>(t);


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

-Advertisement-
Play Games
更多相關文章
  • 一段包含 bytes 類型的 protobuf 二進位數據,經過 pbjs 解碼生成的 json 文件,再傳遞給 pbjs 編碼後生成的二進位數據和原始數據差異巨大,經過一番探究,發現居然是 pbjs 的一個 bug,快來看看你是否踩過這個坑吧~ ...
  • 26種設計模式 轉載:https://zhuanlan.zhihu.com/p/93770973 參考: https://design-patterns.readthedocs.io/zh_CN/latest/read_uml.html https://zhuanlan.zhihu.com/p/93 ...
  • 服務業務線:快遞、快運、中小件、大件、冷鏈、國際、B2B合同物流、CLPS、京喜、三入三出(採購入、退貨入、調撥入、銷售出、退供出、調撥出)等 ...
  • 零售企業的發展路徑 零售企業的發展路徑一般可分為以下幾個階段: 單店經營階段:企業在一個地區或城市開設單個門店。這時,企業需要把精力放在瞭解當地市場和顧客需求上,這是積累經驗和品牌知名度的重要環節。為了在市場中建立競爭力,企業需要不斷提升產品和服務的質量,比如探索新的零售方式、創新商品、提高服務質量 ...
  • 一、引言 本文是京東到家自動化測試體系建設過程中的一些回顧和總結,刪減了部分系統設計與實踐的章節,保留了組織與文化相關的內容,整理成文,以饗讀者。 下麵就以QA(Quality Assurance)的視角來探討工作中經常面臨的問題與挑戰。 關於軟體質量,不知道你有沒有以下困惑: 西醫中“頭疼醫頭,腳 ...
  • 題目:員工工資單計算器 描述: 請編寫一個Python程式,該程式將通過用戶輸入來計算並列印員工的工資單。工資單應該包括員工的姓名、工作時長、每小時工資、毛工資、扣除額和凈工資。扣除額包括稅款和養老金。 要求: 1. 輸入: 員工姓名(字元串) 工作時長(整數,單位:小時) 每小時工資(浮點數,單位 ...
  • 在JDK 21中,Sequenced Collections的引入帶來了新的介面和方法來簡化集合處理。此增強功能旨在解決訪問Java中各種集合類型的第一個和最後一個元素需要非統一且麻煩處理場景。 下麵一起通過本文來瞭解一下不同集合處理示例。 Sequenced Collections介面 Seque ...
  • extern extern 是 C++ 中的一個關鍵字,用於聲明一個變數或函數是在其他文件中定義的。它的作用是告訴編譯器在鏈接時在其他文件中尋找該變數或函數的定義。 在 C++ 中,如果一個變數或函數在多個文件中使用,那麼就需要在每個文件中都聲明一次該變數或函數。這時就可以使用 extern 關鍵字 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...