gpt4 book ai didi

c++ - 在 MFC 回调函数/事件处理程序中捕获异常

转载 作者:行者123 更新时间:2023-12-03 07:50:11 25 4
gpt4 key购买 nike

我有一个带有模式对话框的 MFC 项目。在我的代码的开头,我有一个 try/catch声明,我尝试 throw我的代码的各个地方都有异常。

来自 OnBnClickedButton1 的异常或OnTvnGetdispinfoTree处理得当,但是 throw来自OnTvnSelchangedTree未处理 - 参见下图。

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

主要代码为try/catch声明:

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");
}

throw从最后一个处理程序( SelChanged )不会转到 catch语句,但以错误结束:

Unhandled exception

最佳答案

您不能跨外部堆栈帧抛出 (C++) 异常1TVN_SELCHANGED通知由 TreeView 控件发送到其父级,使其能够响应 UI 状态的更改。当回调返回时,控制权返回到控制实现中。

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

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

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

如果您需要在发生这种情况时收集信息,您可以让安装人员将系统设置为 collect user-mode dumps ,允许您对可能存在错误的内容进行事后分析。


特别关于 TVN_SELCHANGED 的说明:这是一条仅供观察的通知。如果父级决定处理此通知,则处理程序的返回值将被明确忽略。当没有人在听的时候想要尖叫“哦不!”似乎很不寻常。

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


1 Raymond Chen 的博文 When you transfer control across stack frames, all the frames in between need to be in on the joke 中的详细信息.

2 替代方案是 function-try-block 。这更加复杂,因为您必须确保不会“吞掉”您没有预料到的异常。

关于c++ - 在 MFC 回调函数/事件处理程序中捕获异常,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/77370211/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com