在这个基于对话框的应用程序中,GUI会在几秒钟后冻结

GUI freezes after a few seconds in this dialog based application

本文关键字:几秒 冻结 对话框 应用程序 GUI      更新时间:2023-10-16

我是一个初学者,试图理解GUI在几秒钟后冻结的原因。试着运行它,看看我的意思。。难道它不应该永远打印Hello1 Hello2 Hello3 Hello4和Hello5吗?需要明确的是:是SetWindowText()停止了工作,所以是GUI冻结了。我的意思是,应用程序一直在运行,只是"显示文本"部分崩溃了。我知道这一点是因为我在发布之前对它进行了调试。那么我错过了什么?为什么天气很冷?你知道吗?提前感谢

这是main.cpp:

#define WIN32_LEAN_AND_MEAN
#include <windows.h>
#include "resource.h"
using namespace std;
HINSTANCE hInst;
void sayHello(HWND handle)
{
    BOOL working = true;
    while (working)
    {
        SetWindowText(handle, "Hello1");
        SetWindowText(handle, "Hello2");
        SetWindowText(handle, "Hello3");
        SetWindowText(handle, "Hello4");
        SetWindowText(handle, "Hello5");
    }
}
BOOL CALLBACK DialogProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
    switch(uMsg)
    {
        case WM_INITDIALOG:
            return TRUE;
        case WM_CLOSE:
            EndDialog(hwndDlg, 0);
            return TRUE;
        case WM_COMMAND:
            switch(LOWORD(wParam))
            {
                case IDC_BTN_QUIT:
                    EndDialog(hwndDlg, 0);
                    return TRUE;
                case IDC_BTN_TEST:
                    HWND handle = GetDlgItem(hwndDlg, 1003);
                    sayHello(handle);
                    return TRUE;
            }
    }
    return FALSE;
}

int APIENTRY WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nShowCmd)
{
    hInst = hInstance;
    return DialogBox(hInstance, MAKEINTRESOURCE(DLG_MAIN), NULL, (DLGPROC)DialogProc);
}

这是资源。h:

#include <windows.h>
#define DLG_MAIN 101
#define IDC_BTN_TEST 1001
#define IDC_BTN_QUIT 1002
#define ID_CONTROLSTATIC 1003

这里是resource.rc:

#include "resource.h"
DLG_MAIN DIALOGEX 6, 5, 194, 106
CAPTION "Test Application"
BEGIN
  CONTROL "&Test", IDC_BTN_TEST, "Button", 0x10010000, 138,  5, 46, 15
  CONTROL "TEXT HERE", ID_CONTROLSTATIC, "Static", 0x10010000, 35, 35, 50, 8
  CONTROL "&Quit", IDC_BTN_QUIT, "Button", 0x10010000, 138, 29, 46, 15
END

这里有一个无限的looop:

void sayHello(HWND handle)
{
    BOOL working = true;
    while (working)
    {
        SetWindowText(handle, "Hello1");
        SetWindowText(handle, "Hello2");
        SetWindowText(handle, "Hello3");
        SetWindowText(handle, "Hello4");
        SetWindowText(handle, "Hello5");
    }
}

添加退出循环的条件,例如if(something) working = false;

如果你想让它永远打印,你至少需要在命令之间添加一个计时器,因为它打印消息的速度太快了,你不会真正知道它正在发生。

所有windows GUI程序的基本结构都是一个消息循环,类似于:

while (GetMessage(&msg, NULL, 0, 0))
{
    TranslateMessage(&msg);
    DispatchMessage(&msg);
}

程序在GetMessage调用中等待,直到消息到达,然后处理消息(最终在窗口或对话框过程中结束),然后返回GetMessage等待下一条消息。

在您的程序中,消息循环隐藏在对DialogBox的调用中,但它就在那里。

当您单击IDC_BUTTON_TEST时,函数sayHello永远不会返回。因此,控制永远不会返回到消息循环,因此您不再处理任何消息,并且您的程序似乎已挂起。

如果你想与消息循环并行工作,你至少有两个选择:

  1. 创建一个计时器。当您收到WM_TIMER消息时,您需要做一些工作(如显示列表中的下一个字符串),然后返回消息循环并等待更多消息。你在每个计时器上做的工作越多,你的应用程序的响应就越差
  2. 创建另一个线程。这可以在不干扰消息循环的情况下连续运行,但可能会更复杂,因为您必须担心线程之间的同步