MFC MDI 函数调用与 SendMessage C 程序员试图理解一个基本概念

问题描述

作为高级硬件 EE,我的大部分编程时间都根据需要使用 C 进行固件和内联汇编。我一直在研究 MFC MDI,你们中的一些人很好地帮助了我。我正在创建一个设置对话框并从菜单项移开以写入注册表以存储和检索应用程序 user settings。我在星期天用谷歌搜索了六种方法来尝试理解基本的 "Calling a function from another .cpp file class" 和许多其他变体来得到我的答案......失败了(主要是因为我可能不明白他们的答案)。>

我有一个主程序重置,它会破坏我的应用程序创建的 HKEY_CURRENT_USER 键。效果很好,没有问题。现在,我不是将 "Settings" 菜单中的一项与 "Reset App" 项一起移动到 button press 之一中的 property sheet/page dialogs

我让它工作了,但我不清楚为什么我最初关于如何执行它的想法失败了。

所以,在我的应用程序中,我触发了这个函数

void CApplication::OnResetProgramSettings() //Truncated function for StackOF clarity
{    
    // ask the user
    int disableuserWarnings = FALSE;       
    if (disableuserWarnings == FALSE) {

        if (IDNO == AfxMessageBox(IDS_WARNING_RESET_APP,MB_ICONQUESTION | MB_YESNO))
            return;
    }

    LSTATUS status = SHDeleteKey(HKEY_CURRENT_USER,_T("SOFTWARE\\Trains"));    
}

这是从以下调用的:

BEGIN_MESSAGE_MAP(CApplication,CWinAppEx)
   ...other commands
    ON_COMMAND(ID_SETTINGS_RESET_REGISTRY,OnResetProgramSettings)
    ...other commands
    ON_UPDATE_COMMAND_UI(ID_SETTINGS_RESET_REGISTRY,OnUpdateOnResetProgramSettings)
END_MESSAGE_MAP()

为了从我的新 Dialog Settings 部分完成这项工作,我实现了这一点:

void CSettingsReset::OnBnClickedButtonReset()
{
    // Global reset
    AfxGetMainWnd()->SendMessage(WM_COMMAND,ID_SETTINGS_RESET_REGISTRY);

    //CApplication::OnResetProgramSettings();
    //&CApplication::OnResetProgramSettings();
}

就像我选择了上面提到的菜单"Reset App" 一样。

属性页名为 SettingsReset.cpp,上面的内容位于 Application.cpp 中。

在 C 语言中,我通常只调用 application.h 文件中包含的 SettingsReset.h 函数,如下所示:

OnResetProgramSettings();

我意识到这在 C++ 中的工作方式不同。您可以看到从上面的代码中注释掉并在下面取消注释的两个行项目:

    CApplication::OnResetProgramSettings(); //Doesn't get called,static error if memory serves
    &CApplication::OnResetProgramSettings(); //Doesn't get called
    CApplication OnResetProgramSettings; //Creates an object I don't want/need... or do I?
    

所以我向 "achieve" 发送 AFX 消息,结果相同。我的最终问题是 "call" Application.cpp 文件中的 OnResetProgramSettings() FROM SettingsReset.cpp 文件的正确方法是什么?我已经阅读了关于使函数 "Static"内容,但我不清楚如何做到这一点和/或为什么......现在每个函数都有两个单独的处理程序......即OnResetProgramSettings()OnResetProgramSettings1().... 正确吗?

有没有一种方法可以做到这一点,而不必使用 AFX 消息来触发该功能

解决方法

您可以通过两种简单的方法从应用程序类的“外部”调用该类的非静态 成员函数。但它们本质上相同,因为您只需要通过实际应用程序对象(实例)调用该成员函数。

首先,您可以使用该对象本身。在您的代码中的某处,您将拥有一个如下所示的应用程序变量:

CApplication MyApp;

如果您的项目是使用 Visual Studio 向导创建的,那么这将可能在“CApplication.cpp”文件中,并且在“CApplication.cpp”文件中将有一个声明 .h”标题。任何包含该标头的源文件都可以简单地调用 public 成员 函数,如下所示:

MyApp.OnResetProgramSettings();

或者,您可以使用在 MFC 标头中定义的 AfxGetApp() 调用获取指向当前应用程序实例(几乎从任何地方)的指针;但请注意,这必须转换为指向派生类的指针:

static_cast<CApplication*>(AfxGetApp())->OnResetProgramSettings();

然而,像这样调用成员有一个非常重要的警告:你不能对那些在“正常”调用方法中由框架提供的参数的函数执行此操作。因此,例如,它根本不适用于 ON_UPDATE_COMMAND_UI 消息处理程序(它需要 CCmdUI* 参数,由框架提供)。


请注意,消息处理程序在使用类向导创建时被声明为 protected,因此您必须将 CApplication 类稍微更改为将您需要以上述方式访问的内容设为公开。但那是微不足道的。

,

问题在于 ON_UPDATE_COMMAND_UI 宏创建了一个私有(或受保护?)成员函数,在您的情况下为 CApplication::OnResetProgramSettings。这意味着您不能直接从另一个类调用它(在您的情况下为 CSettingsReset)。

一种方法是像您一样通过 PostMessage 触发命令(除非您依赖 AfxGetMainWnd() 而不是在对话框构造函数中获取指向应用程序类的指针 - 如果您以后更改主窗口或使用系统通知图标等)。

另一种方法是将功能移动到您可以直接调用的公共类中,然后让 CApplication::OnResetProgramSettings 简单地调用该新函数。这避免了发送到消息队列的中间步骤。

当然,整个设计都被破坏了,所有这些实用程序都应该从任何特定的窗口类移到一个专用的服务类,该服务类封装了与注册表功能相关的所有内容,并且您的窗口接收并使用指向那个服务类。将其视为过时、糟糕的 MVVM 版本,因为您使用的是过时的技术 (MFC)。