捕获 MFC 回调函数/事件处理程序中的异常

Catch exception inside MFC callback function / event handlers

提问人:Michal Hadraba 提问时间:10/27/2023 最后编辑:Remy LebeauMichal Hadraba 更新时间:10/27/2023 访问量:127

问:

我有一个带有模式对话框的 MFC 项目。在我的代码开头,我有一个 / 语句,我尝试在代码的各个位置出现异常。trycatchthrow

异常 from or 已正确处理,但 from 未处理 - 见下图。OnBnClickedButton1OnTvnGetdispinfoTreethrowOnTvnSelchangedTree

它是由 Visual Studio 2022 向导生成的基于对话框的标准 MFC 项目。对话框上有一个按钮和 .CTreeControl

带 / 语句的主要代码:trycatch

try {
  CMFCExceptDlg dlg;
  m_pMainWnd = &dlg;
  INT_PTR nResponse = dlg.DoModal();
  if (nResponse == IDOK)  {
     TRACE(traceAppMsg, 0, "OK...\n");
  }
  else if (nResponse == IDCANCEL)  {
     TRACE(traceAppMsg, 0, "Cancel...\n");
  }
  else if (nResponse == -1)  {
      TRACE(traceAppMsg, 0, "Warning: dialog creation failed, so application is 
     terminating unexpectedly.\n");
      TRACE(traceAppMsg, 0, "Warning: if you are using MFC controls on the dialog, you 
      cannot #define _AFX_NO_MFC_CONTROLS_IN_DIALOGS.\n");
  }   
 }    
 catch(std::exception &exc)   {
    const char *x = exc.what();
    TRACE(traceAppMsg, 0, exc.what());
 }

对话框代码(.h 文件):

class CMFCExceptDlg : public CDialogEx 
{
public:
    CMFCExceptDlg(CWnd* pParent = nullptr); // standard constructor

#ifdef AFX_DESIGN_TIME
    enum { IDD = IDD_MFC_EXCEPT_DIALOG };
#endif

protected:
    virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV support

protected:
    HICON m_hIcon;
    CTreeCtrl m_cTree;
    virtual BOOL OnInitDialog();
    DECLARE_MESSAGE_MAP()
public:
    afx_msg void OnTvnGetdispinfoTree(NMHDR *pNMHDR, LRESULT *pResult);
    afx_msg void OnBnClickedButton1();
    afx_msg void OnTvnSelchangedTree(NMHDR* pNMHDR, LRESULT* pResult);
};

对话的实现 (.cpp):

CMFCExceptDlg::CMFCExceptDlg(CWnd* pParent /*=nullptr*/)
: CDialogEx(IDD_MFC_EXCEPT_DIALOG, pParent)
{
    m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);
}

void CMFCExceptDlg::DoDataExchange(CDataExchange* pDX)
{
    CDialogEx::DoDataExchange(pDX);
    DDX_Control(pDX, IDC_TREE, m_cTree);
}

BEGIN_MESSAGE_MAP(CMFCExceptDlg, CDialogEx)

    ON_NOTIFY(TVN_GETDISPINFO, IDC_TREE, &CMFCExceptDlg::OnTvnGetdispinfoTree)
    ON_BN_CLICKED(IDC_BUTTON1, &CMFCExceptDlg::OnBnClickedButton1)
    ON_NOTIFY(TVN_SELCHANGED, IDC_TREE, &CMFCExceptDlg::OnTvnSelchangedTree)
END_MESSAGE_MAP()

// CMFCExceptDlg message handlers
BOOL CMFCExceptDlg::OnInitDialog()
{
    CDialogEx::OnInitDialog();

    m_cTree.InsertItem(LPSTR_TEXTCALLBACK, TVI_ROOT);
    m_cTree.InsertItem(LPSTR_TEXTCALLBACK, TVI_ROOT);
    m_cTree.InsertItem(LPSTR_TEXTCALLBACK, TVI_ROOT);
    m_cTree.InsertItem(LPSTR_TEXTCALLBACK, TVI_ROOT);

    return TRUE;  
}

void CMFCExceptDlg::OnTvnGetdispinfoTree(NMHDR *pNMHDR, LRESULT *pResult)
{
    LPNMTVDISPINFO pTVDispInfo = reinterpret_cast<LPNMTVDISPINFO>(pNMHDR);
    *pResult = 0;
    throw std::exception("GetDispInfo");
}

void CMFCExceptDlg::OnBnClickedButton1()
{
    throw std::exception("In Click Button Error");
}

void CMFCExceptDlg::OnTvnSelchangedTree(NMHDR* pNMHDR, LRESULT* pResult)
{
    LPNMTREEVIEW pNMTreeView = reinterpret_cast<LPNMTREEVIEW>(pNMHDR);
    // TODO: Add your control notification handler code here
    *pResult = 0;
    throw std::exception("OnTvnSelchangedTree");
}

from this last 处理程序 () 不会转到语句,而是以错误结束:throwSelChangedcatch

Unhandled exception

C++ 异常 WinAPI MFC 处理程序

评论

0赞 273K 10/27/2023
这是从窗口消息循环调用的,你没有显示它,它有try-catch吗?
0赞 Ahmed AEK 10/27/2023
我对显示的调用堆栈感兴趣,它是否包括您的 try 块?
1赞 Remy Lebeau 10/27/2023
@273K您引用的消息循环位于 MFC 的DoModal()
1赞 RbMm 10/27/2023
这与有没有办法让 DisableUserModeCallbackFilter 在 Windows 10 中工作?相同。你得到了STATUS_FATAL_USER_CALLBACK_EXCEPTION
3赞 IInspectable 10/27/2023
不能在外部帧之间抛出 (C++) 异常。您需要找到不同的解决方案。

答:

5赞 IInspectable 10/27/2023 #1

不能在外部堆栈帧1 之间抛出 (C++) 异常。树视图控件将TVN_SELCHANGED通知发送到其父级,使其能够响应 UI 状态的更改。当回调返回时,控制权将交叉回控件实现。

控件未为 C++ 异常做好准备。据我们所知,它可以用 C 语言实现,甚至不知道 C++ 异常是什么。因此,至关重要的是,C++ 异常永远不会逃脱窗口过程实现。

确保不会发生这种情况的最简单方法2 是将 noexcept 说明符应用于窗口过程。由于这是 MFC,因此您不控制窗口过程,而是必须将所有消息处理程序标记为 。这样,任何跨消息处理程序引发 C++ 异常的尝试都会让语言运行时启动受控的紧急关闭。noexcept

在所有结果中,这是您所能希望的最好的结果。

如果发生这种情况时需要收集信息,可以让安装程序设置系统以收集用户模式转储,从而允许您对很可能是 bug 的内容进行事后分析。


具体说明:这是仅供观察的通知。如果父级决定处理此通知,则显式忽略处理程序的返回值。在没有人听的时候,想要尖叫“哦,不!”似乎是很不寻常的。TVN_SELCHANGED

如果您想取消选择,则需要改为回复TVN_SELCHANGING消息。观察其返回值并控制是否允许选择更改。


1 Raymond Chen 博客文章中的细节 当您在堆栈帧之间转移控制权时,中间的所有帧都需要在笑话中

2 另一种选择是 function-try-block。这更复杂,因为你必须确保你不会“吞下”你意想不到的异常。

评论

0赞 RbMm 10/27/2023
C++ 例外与否 C++ 这里无关紧要。如果窗口消息回调中出现任何异常,结果将相同
0赞 IInspectable 10/27/2023
@RbMm 没错。这就是为什么介绍性句子读作“你不能在外部堆栈帧之间抛出(C++)异常”,而不是“你不能在外部堆栈帧之间抛出C++异常”。
0赞 Michal Hadraba 10/27/2023
我理解。但是,为什么 IT 在其他回调函数中工作呢?正如我在问题中写的那样?例如。ClickButton(它是不同的控件)或 GetDispInfo(它是同一控件)
2赞 IInspectable 10/27/2023
@MichalHadraba 它似乎有效,因为它(目前)没有失败。我不满足于弄清楚系统在超出规格使用时如何响应。如果您对此感兴趣,请 ping RbMm。他会发现为什么事情会像他的系统一样。如果你真的很好奇,但当你想发布保证工作的软件时,这很有用。