Skip to content

SetWindowLongPtr and GetWindowLongPtr 用法 实例(迁移2011-01-28)

Published: at 02:52 PM | 4 min read

SetWindowLongPtr与GetWindowLongPtr主要有两种用法:
第一种是:改变指定窗口的属性;
第二种是:设置一个值在额外存储空间的指定偏移位置。

下面所说的是第二种用法:
首先,额外存储空间是指在设计一个窗口类时的cbWndExtra属性, 如:

wndclass.cbWndExtra = 2 * sizeof(size_t);//额外分配两个size_t的空间

用SetWindowLongPtr可以在这个空间的某个偏移位置设置一个值, 当然设置一个值用处不大, 通常在这里设置的是一个地址, 这个地址可以是某个结构体的地址, 这样存储的数据就多了, 如:

//GWL_EXTRAMEMORY 额外内存里的偏移位置, hExtra存储的地址
SetWindowLongPtr(hwnd, GWL_EXTRAMEMORY, reinterpret_cast<LONG_PTR>(hExtra));

用GetWindowLongPtr获取上面所设置的值, 如:

hExtra = (HGLOBAL)GetWindowLongPtr(hwnd, GWL_EXTRAMEMORY);

下面简单例子是展示它们的用法, 但是并不能体现出真正的用处。
真正的用处是:当分配的内存在堆上时, 我们可以通过GetWindowLongPtr随时访问这块内存, 为了防止造成不必要的错误, 访问时最好加锁。

#include <windows.h>

#define GWL_EXTRAMEMORY sizeof(size_t)

typedef struct tagExtraMemory EXTRAMEMORY;
typedef struct tagExtraMemory* HEXTRAMEMORY;
struct tagExtraMemory
{
	size_t	num;
	wchar_t name[32];
};

LRESULT CALLBACK WindowProc(HWND, UINT, WPARAM, LPARAM);


int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance,
					 LPSTR lpCmdLine, int nCmdShow)
{
	static wchar_t szAppName[] = L"ExtraMemory";
	HWND hwnd;
	MSG msg;
	WNDCLASS wndclass;

	wndclass.cbClsExtra = 0;
	wndclass.cbWndExtra = 2 * sizeof(size_t);//额外分配两个size_t的空间
	wndclass.hbrBackground = (HBRUSH)GetStockObject(WHITE_BRUSH);
	wndclass.hCursor = LoadCursor(NULL, IDC_ARROW);
	wndclass.hIcon = LoadIcon(NULL, IDI_APPLICATION);
	wndclass.hInstance = hInstance;
	wndclass.lpfnWndProc = WindowProc;
	wndclass.lpszClassName = szAppName;
	wndclass.lpszMenuName = NULL;
	wndclass.style = CS_HREDRAW | CS_VREDRAW;
	
	if (!RegisterClass(&wndclass))
	{
		MessageBox(NULL, L"Registerclass fail", L"tips", MB_ICONERROR);
		return 0;
	}

	hwnd = CreateWindow(szAppName, L"tjw", WS_OVERLAPPEDWINDOW,
		CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT,
		NULL, NULL, hInstance, NULL);

	ShowWindow(hwnd, nCmdShow);
	UpdateWindow(hwnd);

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

	return (int)msg.wParam;
}

LRESULT CALLBACK WindowProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
{
	HDC hdc;
	PAINTSTRUCT ps;
	RECT rect;

	HGLOBAL hExtra;
	LPVOID lpExtra;

	switch(message)
	{
	case WM_CREATE:
		{
			hExtra  = GlobalAlloc(GHND, sizeof(EXTRAMEMORY));//分配堆内存
			lpExtra = GlobalLock(hExtra);//在这里打断点, 查看hExtra的内存地址
			SetWindowLongPtr(hwnd, GWL_EXTRAMEMORY, reinterpret_cast<LONG_PTR>(hExtra));
			GlobalUnlock(hExtra);
		}
		break;

	case WM_PAINT:
		hdc = BeginPaint(hwnd, &ps);

		GetClientRect(hwnd, &rect);
		DrawText(hdc, L"hello, world!", -1, &rect,
			DT_SINGLELINE | DT_CENTER | DT_VCENTER);

		EndPaint(hwnd, &ps);
		break;

	case WM_DESTROY:
		{
			hExtra = (HGLOBAL)GetWindowLongPtr(hwnd, GWL_EXTRAMEMORY);
			lpExtra = GlobalLock(hExtra);//在这里打断点, 查看hExtra的内存地址
			GlobalUnlock(hExtra);
			GlobalFree(hExtra);//释放堆内存
			
			PostQuitMessage(0);
		}
		break;
	}
	return DefWindowProc(hwnd, message, wParam, lParam);
}

注意:偏移位置加上指针的字节数不能超过额外存储空间 如:32位系统额外存储空间为8, 偏移位置位4, 再加上指针是4个字节, 这样正好; 如果是64位系统, 指针是8个字节, 所以一定要注意。