MSDN原文(英文)

管理应用程序状态

一个窗口过程仅仅是一个为每个消息获取调用函数,所以它本质上是无状态的。因此,你需要一个方法来跟踪你的应用程序从一个函数调用下一个函数的状态。
最简单的方法是把一切都放在全局变量中。这对于小程序已经足够了,并且许多SDK示例都使用这种方式。然而在一个大型程序,它会导致全局变量的扩散。此外,你可能有几个窗口,每个都有其自己的窗口过程,跟踪哪个窗口应该访问哪些变量变得混乱和易出错。

CreateWindowEx函数提供了一种方法可以将任何数据结构传递给一个窗口,当这个函数被调用,以下两个消息发送到你。
  • WM_NCCREATE
  • WM_CREATE
这些消息按列出的顺序发送(在CreateWindowEx期间这发送的不止这两个消息,但我们在这个讨论里可以忽略)。

WM_NCCREATE和WM_CREATE消息在窗口可见前发送,这是制定和初始化你的UI的好地方——例如,确定窗口的初始布局。


函数CreateWindowEx的最后一个参数是void*类型的指针,在这个参数你可以传递任何你想要的指针值,当窗口过程处理WM_NCCREATE和WM_CREATE消息,它可以从消息数据中提取这个值。

让我们看看如何使用这个参数将应用程序数据传递到你窗口。首先定义一个类或结构保存状态信息:

// 定义一个结构保存一些状态信息

struct StateInfo {

    // ... (结构成员没有显示)

};

当你调用CreateWindowsEx函数,将结构指针传递给最后的void*参数。

StateInfo *pState = new (std::nothrow) StateInfo;

    if (pState == NULL)
{
return 0;
} // 在这里初始化结构成员 (示例没有显示). HWND hwnd = CreateWindowEx(
0, // Optional window styles.
CLASS_NAME, // Window class
L"Learn to Program Windows", // Window text
WS_OVERLAPPEDWINDOW, // Window style // Size and position
CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, NULL, // Parent window
NULL, // Menu
hInstance, // Instance handle
pState // 额外的数据信息
);

当你收到WM_NCCREATE和WM_CREATE消息,每个消息的 lParam的参数指向CREATESTRUCT结构的指针。而后,CREATESTRUCT结构包含的指针传入CreateWindowEx。

图示CREATESTRUCT结构布局
 
以下是如何提取数据结构的指针,首先,通过计算lParam参数获得CREATESTRUCT结构。
CREATESTRUCT *pCreate = reinterpret_cast<CREATESTRUCT*>(lParam);
CREATESTRUCT

结构

lpCreateParams

成员


CreateWindowEx


指定

原始

void

指针
。计算lpCreateParams获取数据结构的指针。

pState = reinterpret_cast<StateInfo*>(pCreate->lpCreateParams);
下一步,调用 SetWindowLoingPtr函数并把指针传入你的数据结构。
SetWindowLongPtr(hwnd, GWLP_USERDATA, (LONG_PTR)pState);

最后一个函数的调用的目的是将StateInfo指针储存在窗口的实例数据中。一旦你这样做,你总是可以从窗口调用
GetWindowLongPtr得到指针。

LONG_PTR ptr = GetWindowLongPtr(hwnd, GWLP_USERDATA);
StateInfo *pState = reinterpret_cast<StateInfo*>(ptr);

每个窗口都有自己的实例数据,所以你可以创建多个窗口,每个窗口都有自己的数据结构的实例。这种方法是特别有用的,如果你定义一个窗口类和创建该类的多个窗口。比如你创建了一个自定义控件类。GetWindowsLongPtr调用可以很方便的封装在一个小的辅助函数。

inline StateInfo* GetAppState(HWND hwnd)
{
LONG_PTR ptr = GetWindowLongPtr(hwnd, GWLP_USERDATA);
StateInfo *pState = reinterpret_cast<StateInfo*>(ptr);
return pState;
}

现在你可以写你的窗口过程如下:

LRESULT CALLBACK WindowProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
StateInfo *pState;
if (uMsg == WM_CREATE)
{
CREATESTRUCT *pCreate = reinterpret_cast<CREATESTRUCT*>(lParam);
pState = reinterpret_cast<StateInfo*>(pCreate->lpCreateParams);
SetWindowLongPtr(hwnd, GWLP_USERDATA, (LONG_PTR)pState);
}
else
{
pState = GetAppState(hwnd);
} switch (uMsg)
{ // Remainder of the window procedure not shown ... }
return TRUE;
}

面向对象的方法

 
我们可以将这种方法进一步。我们已经定义了一个数据结构来保持窗口状态信息。 提供 此 数据 结构 的 成员 操作 的 函数 (方法) 的 数据是 有 意义的。这自然会使得一个结构(或类)的设计负责窗口的所有操作。窗口过程会成为类的一部分。
换句话说,我们要从这走:
// 伪代码

LRESULT CALLBACK WindowProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
StateInfo *pState; /* Get pState from the HWND. */ switch (uMsg)
{
case WM_SIZE:
HandleResize(pState, ...);
break; case WM_PAINT:
HandlePaint(pState, ...);
break; // And so forth.
}
}
到这:
// 伪代码

LRESULT MyWindow::WindowProc(UINT uMsg, WPARAM wParam, LPARAM lParam)
{
switch (uMsg)
{
case WM_SIZE:
this->HandleResize(...);
break; case WM_PAINT:
this->HandlePaint(...);
break;
}
}

唯一的问题是用什么方法连接MyWindow::WindowProc。RegisterClass函数需要的窗口过程是一个函数指针,在此上下文中,你不能将指针传递给(非静态)成员函数。然而,你可以将指针传递给一个静态成员函数,然后委托给成员函数。以下是一个类模板,显示这种方法:

template <class DERIVED_TYPE>
class BaseWindow
{
public:
static LRESULT CALLBACK WindowProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
DERIVED_TYPE *pThis = NULL; if (uMsg == WM_NCCREATE)
{
CREATESTRUCT* pCreate = (CREATESTRUCT*)lParam;
pThis = (DERIVED_TYPE*)pCreate->lpCreateParams;
SetWindowLongPtr(hwnd, GWLP_USERDATA, (LONG_PTR)pThis); pThis->m_hwnd = hwnd;
}
else
{
pThis = (DERIVED_TYPE*)GetWindowLongPtr(hwnd, GWLP_USERDATA);
}
if (pThis)
{
return pThis->HandleMessage(uMsg, wParam, lParam);
}
else
{
return DefWindowProc(hwnd, uMsg, wParam, lParam);
}
} BaseWindow() : m_hwnd(NULL) { } BOOL Create(
PCWSTR lpWindowName,
DWORD dwStyle,
DWORD dwExStyle = 0,
int x = CW_USEDEFAULT,
int y = CW_USEDEFAULT,
int nWidth = CW_USEDEFAULT,
int nHeight = CW_USEDEFAULT,
HWND hWndParent = 0,
HMENU hMenu = 0
)
{
WNDCLASS wc = {0}; wc.lpfnWndProc = DERIVED_TYPE::WindowProc;
wc.hInstance = GetModuleHandle(NULL);
wc.lpszClassName = ClassName(); RegisterClass(&wc); m_hwnd = CreateWindowEx(
dwExStyle, ClassName(), lpWindowName, dwStyle, x, y,
nWidth, nHeight, hWndParent, hMenu, GetModuleHandle(NULL), this
); return (m_hwnd ? TRUE : FALSE);
} HWND Window() const { return m_hwnd; } protected: virtual PCWSTR ClassName() const = 0;
virtual LRESULT HandleMessage(UINT uMsg, WPARAM wParam, LPARAM lParam) = 0; HWND m_hwnd;
};

BaseWindow是一个抽象类,从特定的窗口了派生。例如,以下是一个简单的派生类的basewindow的声明:

class MainWindow : public BaseWindow<MainWindow>
{
public:
PCWSTR ClassName() const { return L"Sample Window Class"; }
LRESULT HandleMessage(UINT uMsg, WPARAM wParam, LPARAM lParam);
};

要创建一个窗口,调用BaseWindow::Create:

int WINAPI wWinMain(HINSTANCE hInstance, HINSTANCE, PWSTR pCmdLine, int nCmdShow)
{
MainWindow win; if (!win.Create(L"Learn to Program Windows", WS_OVERLAPPEDWINDOW))
{
return 0;
} ShowWindow(win.Window(), nCmdShow); // Run the message loop. MSG msg = { };
while (GetMessage(&msg, NULL, 0, 0))
{
TranslateMessage(&msg);
DispatchMessage(&msg);
} return 0;
}

纯虚拟的 BaseWindow::HandleMessage方法用于执行窗口过程。下面的执行是相当于开始显示窗口过程:

LRESULT MainWindow::HandleMessage(UINT uMsg, WPARAM wParam, LPARAM lParam)
{
switch (uMsg)
{
case WM_DESTROY:
PostQuitMessage(0);
return 0; case WM_PAINT:
{
PAINTSTRUCT ps;
HDC hdc = BeginPaint(m_hwnd, &ps);
FillRect(hdc, &ps.rcPaint, (HBRUSH) (COLOR_WINDOW+1));
EndPaint(m_hwnd, &ps);
}
return 0; default:
return DefWindowProc(m_hwnd, uMsg, wParam, lParam);
}
return TRUE;
}

注意,窗口句柄存储在成员变量(m_hwnd)所以我们不需要将它作为一个参数传递给HandleMessage。

许多现有的Windows编程框架,如Microsoft基础类(MFC)和活动模板库(ATL)使用的基本上类似于此处所示的方法。当然,一个完全广义的框架(如MFC)比这种相对简单的例子更复杂。

你的第一Windows程序——管理应用程序状态的更多相关文章

  1. 0x02 译文:Windows桌面应用Win32第一个程序

    本节课我们将用C++ 写一个最简单的Windows 程序. 目录: 创建一个窗口 窗口消息 编写窗口过程 绘制窗口 关闭窗口 管理应用程序状态 代码如下: #ifndef UNICODE #defin ...

  2. SharePoint 创建列表并使用Windows Presentation Foundation应用程序管理列表

    SharePoint创建列表并使用程序管理列表         列表是SharePoint开发者输入数据的方式之中的一个.使用Web界面创建一个列表并加入一些数据.过程例如以下: 1. 打开站点. 2 ...

  3. Windows程序内部运行机制 转自http://www.cnblogs.com/zhili/p/WinMain.html

    一.引言 要想熟练掌握Windows应用程序的开发,首先需要理解Windows平台下程序运行的内部机制,然而在.NET平台下,创建一个Windows桌面程序,只需要简单地选择Windows窗体应用程序 ...

  4. 深入浅出话VC++(1)——Windows程序内部运行机制

    一.引言 要想熟练掌握Windows应用程序的开发,首先需要理解Windows平台下程序运行的内部机制,然而在.NET平台下,创建一个Windows桌面程序,只需要简单地选择Windows窗体应用程序 ...

  5. CentOS学习笔记--程序管理

    程序管理 一个程序被加载到内存当中运行,那么在内存内的那个数据就被称为程序(process).程序是操作系统上非常重要的概念, 所有系统上面跑的数据都会以程序的型态存在.那么系统的程序有哪些状态?不同 ...

  6. Windows内存管理[转]

    本文主要内容:1.基本概念:物理内存.虚拟内存:物理地址.虚拟地址.逻辑地址:页目录,页表2.Windows内存管理3.CPU段式内存管理4.CPU页式内存管理 一.基本概念1. 两个内存概念物理内存 ...

  7. windows程序员进阶系列:《软件调试》之堆 (一)

    windows程序员进阶系列:<软件调试>之堆 (一) 堆是软件在运行时动态申请内存空间的主要途径.从堆上申请来的空间需要程序员自己申请和释放,且申请和释放操作必须绝对匹配.忘记释放或者多 ...

  8. 第十七章、程序管理与 SELinux 初探

    ---恢复内容开始--- 什么是程序 (process) 在 Linux 底下所有的命令与你能够进行的动作都与权限有关, 而系统依据UID/GID以及文件的属性相关性判定你的权限!在 Linux 系统 ...

  9. windows虚拟内存管理

    内存管理是操作系统非常重要的部分,处理器每一次的升级都会给内存管理方式带来巨大的变化,向早期的8086cpu的分段式管理,到后来的80x86 系列的32位cpu推出的保护模式和段页式管理.在应用程序中 ...

随机推荐

  1. java transient关键字和transaction的区别

    transient:表示临时的,不会被持久化,保存进数据库 transaction:表示事务 <div style="background: #fff; color: #0ff;&qu ...

  2. android 广播分类

    安卓广播分为两类:1.普通广播, broadcast,广播发出之后所有满足条件的应用都能获取到广播里面的数据,缺点是应用获取广播中的数据修改之后不能传递给其它接收广播的应用:2.有序广播,orderb ...

  3. 启发式算法、寻路算法A*算法

    原文链接: http://blog.csdn.net/b2b160/article/details/4057781

  4. Mysql 视图笔记

    1.       视图的定义 视图就是从一个或多个表中,导出来的表,是一个虚拟存在的表.视图就像一个窗口(数据展示的窗口),通过这个窗口,可以看到系统专门提供的数据(也可以查看到数据表的全部数据),使 ...

  5. SpringMVC 文件上传配置,多文件上传,使用的MultipartFile(转)

    文件上传项目的源码下载地址:http://download.csdn.net/detail/swingpyzf/6979915   一.配置文件:SpringMVC 用的是 的MultipartFil ...

  6. 由浅入深完全掌握Ajax

    n年前,如果不知道 XML,您就是一只无人重视的丑小鸭.十八个月前,Ruby 成了关注的中心,不知道 Ruby 的程序员只能坐冷板凳了.今天,如果想跟上最新的技术时尚,那您的目标就是 Ajax. 但是 ...

  7. C#中串口与Modem的通信

    C#中串口与Modem的通信 2007-08-20 09:52643人阅读评论(8)收藏举报 最近一段时间,试验了串口的数据传输.在C#中,其实有一个很好的类SerialPort使串口间的通信变得简单 ...

  8. js实现数组内元素随机排序

    其实蛮容易实现的,关键是简洁与否,下面是我自己写的. function randomSort(a){ var arr = a, random = [], len = arr.length; for ( ...

  9. HTML5最佳实践

    首先先给大家推荐个不错的 前端 网站:http://www.tystudio.net HTML5正迅速称为web前端开发技术标准,作为一名前段开发人员,了解并正确的使用HTML5制作网站变得越来越重要 ...

  10. java eclipse 连接数据库全过程

    1.需要下载一个jar包.地址 http://pan.baidu.com/s/1i50LRId 2.代码如下: import java.sql.*; public class Mytest { pub ...