c++ - 如何定义与使用 lambda 兼容的函数指针并将捕获作为回调

标签 c++ c++11 lambda visual-studio-2015 mfc

需要一个带有 MFC 项目的简单模式对话框的几个变体,我编写了一个简单的类 CDialogDlg,它扩展了标准 MFC CDialog 类并包含用于实现的 Hook 通常在扩展 CDialog 的类中作为方法实现的特定回调。这些回调(如果提供)用于处理某些事件,例如对话框初始化、OK 按钮处理和 DoDataExchange() 函数,该函数提供连接对话框类变量和实际对话框控件的方法。

我的此类的第一个版本使用标准函数进行回调。所以扩展类的回调设置方法设置了一个函数指针,指向标准函数的地址。

void SetDataExchangeCallBack(void(*f)(CDataExchange* pDX)) { funcDX = f; };
void SetInitCallBack(void(*f)(CWnd *dlgWnd)) { funcInit = f; }
void SetOnOkCallBack(void(*f)(CWnd *dlgWnd, CDocument *pDoc)) { funcOK = f; }

然后我意识到我应该能够使用 lambda 而不是标准函数。第一个版本的 lambda 使用与标准函数相同的参数,没有捕获任何编译好的变量,并且可以很好地与扩展类中的现有方法和函数指针一起工作。

但是,当我尝试在回调设置方法中指定的 lambda 中捕获变量时,我遇到了编译错误。

审查 Passing capturing lambda as function pointerC++ lambda with captures as a function pointer我知道尝试使用捕获来执行 lambda 不会使用我正在使用的函数指针和回调设置函数的定义进行编译。

所以我决定我应该在类中添加一个额外的 lambda 特定函数指针以及覆盖现有方法,并使用一个额外的方法在类中使用 lambda 特定函数指针设置回调。

问题:设置回调函数的函数指针定义和参数定义应该是什么样的?我想在方法 CPCSampleView::OnDisplayReportList () 中捕获局部变量 CPCSampleDoc *pDoc。此变量包含指向我想捕获的 View 的 CDocument 派生对象的指针,而不是使用存储它的方法并执行 static_cast 将其取回未捕获的 lambda 的当前版本。

源代码

扩展 CDialogCDialogDlg 类如下所示:

class CDialogDlg : public CDialog
{
    // Construction
    void(*funcDX)(CDataExchange* pDX);
    void(*funcDXpDoc)(CDataExchange* pDX, CDocument *pDoc);
    void(*funcInit)(CWnd *dlgWnd);
    void(*funcOK)(CWnd *dlgWnd, CDocument *pDoc);
    CDocument *m_pDoc;

public:
    CDialogDlg(UINT nIDTemplate, CWnd* pParentWnd = NULL, void(*f)(CDataExchange* pDX) = NULL) : CDialog(nIDTemplate, pParentWnd) { funcDX = f; funcInit = NULL; }
    CDialogDlg(UINT nIDTemplate, CDocument *pDoc, CWnd* pParentWnd = NULL, void(*f)(CDataExchange* pDX, CDocument *pDoc) = NULL) : CDialog(nIDTemplate, pParentWnd) {
        funcDX = NULL; funcDXpDoc = f; funcInit = NULL; m_pDoc = pDoc;
    }

    void SetDataExchangeCallBack(void(*f)(CDataExchange* pDX)) { funcDX = f; };
    void SetInitCallBack(void(*f)(CWnd *dlgWnd)) { funcInit = f; }
    void SetOnOkCallBack(void(*f)(CWnd *dlgWnd, CDocument *pDoc)) { funcOK = f; }

    // Dialog Data
    //{{AFX_DATA(CCashierNoDlg)
    //}}AFX_DATA

    // Overrides
    // ClassWizard generated virtual function overrides
    //{{AFX_VIRTUAL(CCashierNoDlg)
protected:
    virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV support
    //}}AFX_VIRTUAL

protected:
    // Generated message map functions
    //{{AFX_MSG(CCashierNoDlg)
    virtual BOOL OnInitDialog();
    virtual void OnOK();
    //}}AFX_MSG
    DECLARE_MESSAGE_MAP()
};

BEGIN_MESSAGE_MAP(CDialogDlg, CDialog)
    //{{AFX_MSG_MAP(CCashierNoDlg)
    //}}AFX_MSG_MAP
END_MESSAGE_MAP()

void CDialogDlg::DoDataExchange(CDataExchange* pDX)
{
    CDialog::DoDataExchange(pDX);
    if (funcDX) funcDX(pDX);
    if (funcDXpDoc) funcDXpDoc(pDX, m_pDoc);
    //{{AFX_DATA_MAP(CCashierNoDlg)
    //}}AFX_DATA_MAP
}

BOOL CDialogDlg::OnInitDialog()
{
    CDialog::OnInitDialog();
    if (funcInit) funcInit(this);
    return TRUE;  // return TRUE unless you set the focus to a control
                  // EXCEPTION: OCX Property Pages should return FALSE
}

void CDialogDlg::OnOK()
{
    if (funcOK) funcOK(this, m_pDoc);
    CDialog::OnOK();
}

由菜单选择触发的 CView 方法显示一个对话框,其中包含可供选择的项目列表。显示的对话框是一个具有特定对话框模板 ID 的 CDialogDlg 对象。对于特殊处理,我使用了两个不同的回调,它们当前使用的是不捕获的 lambda。结果如下:

void CPCSampleView::OnDisplayReportList ()
{
    CPCSampleDoc *pDoc = GetDocument();

    CDialogDlg myDialog(IDD_DIALOG_REPORTLIST, pDoc, this, [](CDataExchange* pDX, CDocument *pDoc) {
        if (pDX->m_bSaveAndValidate) {
        }
        else {
            CPCSampleDoc *pDocDoc = static_cast<CPCSampleDoc *>(pDoc);
            POSITION pos = NULL;

            do {
                CPCSampleDoc::ListReportList sectionHeader;
                CListBox *x = static_cast<CListBox *>(pDX->m_pDlgWnd->GetDlgItem(IDC_LIST1));

                pos = pDocDoc->GetReportSectionHeader(pos, sectionHeader);
                x->AddString(sectionHeader.m_SectionTitle);
            } while (pos);
        }
    });

    myDialog.SetOnOkCallBack([](CWnd *dlgWnd, CDocument *pDoc) {
        CPCSampleDoc *pDocDoc = static_cast<CPCSampleDoc *>(pDoc);
        CListBox *x = static_cast<CListBox *>(dlgWnd->GetDlgItem(IDC_LIST1));
        int iPtr = x->GetCurSel();
        POSITION pos = NULL;

        do {
            CPCSampleDoc::ListReportList sectionHeader;

            pos = pDocDoc->GetReportSectionHeader(pos, sectionHeader);
            if (iPtr < 1) {
                pDocDoc->MoveToReportSectionHeader(sectionHeader.m_ListOffset);
                break;
            }
            iPtr--;
        } while (pos);
    });

    myDialog.DoModal();
}

最佳答案

您可能会使用 std::function允许捕获 lambda 和其他仿函数以及常规函数指针:

void(*funcDX)(CDataExchange* pDX);
void(*funcDXpDoc)(CDataExchange* pDX, CDocument *pDoc);
void(*funcInit)(CWnd *dlgWnd);
void(*funcOK)(CWnd *dlgWnd, CDocument *pDoc);

成为

std::function<void(CDataExchange*)> funcDX;
std::function<void(CDataExchange*, CDocument*)> funcDXpDoc;
std::function<void(CWnd*)> funcInit;
std::function<void(CWnd*, CDocument*)> funcOK;

你的 setter/constructor 也应该将指针函数更改为 std::function:

void SetInitCallBack(std::function<void(CWnd*)> f) { funcInit = f; }

否则你的用法是相同的:

if (funcDX) funcDX(pDX);

funcInit = nullptr;

关于c++ - 如何定义与使用 lambda 兼容的函数指针并将捕获作为回调,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57897457/

相关文章:

c++ - TensorFlow图像识别输出格式c++

c++ - 模板类具有循环依赖

c++ - 如何从函数返回 std::seed_seq 对象

c# - 在 .Net 中设置/删除事件处理程序

java - lambda 中的隐含匿名类型

c++ - 如何为实例提供 C 库中的函数?

c++ - 为什么无论类型如何,类中的引用变量总是占用 4 个字节? (在 32 位系统上)

c++ - 创建 unordered_set 的 unordered_set

c++ - 是否可以对每个参数进行可变参数宏替换?

java - 如何在 Kotlin 中组合 lambda