我正在使用BitBlt在我的按钮上显示位图。对于大多数情况,它很好,但有一个内存泄漏导致程序在一段时间后崩溃。我做错了什么?
int Springboard::DrawBasicButtons(DRAWITEMSTRUCT* pdis, HINSTANCE hInstance){
RECT rect;
static HBITMAP hCurrIcon, hIconoff, hIconon;
rect = pdis->rcItem;
HFONT font = CreateFont(13, 0, 0, 0, 300, FALSE, FALSE, FALSE, DEFAULT_CHARSET, OUT_DEFAULT_PRECIS, CLIP_DEFAULT_PRECIS, DEFAULT_QUALITY, DEFAULT_PITCH, L"Arial");
TCHAR Txtstr[MAX_PATH];
BOOL isText = FALSE;
int textsize;
if (IDC_HOLD == pdis->CtlID) {
hIconoff = (HBITMAP) LoadBitmap(hInstance, MAKEINTRESOURCE(BASIC_HOLDOFF));
hIconon = (HBITMAP) LoadBitmap(hInstance, MAKEINTRESOURCE(BASIC_HOLDON));
_tcscpy( Txtstr, _T("Hold "));
isText = TRUE;
if (pdis->itemState & ODS_SELECTED) hCurrIcon = hIconon;
else hCurrIcon = hIconoff;
}
HDC hdc = CreateCompatibleDC(pdis->hDC);
SelectObject(hdc, hCurrIcon);
BitBlt(pdis->hDC,0, 0,ICON_WIDTH,ICON_HEIGHT, hdc, 0, 0, SRCCOPY);
if(isText == TRUE){
textsize = _tcslen(Txtstr);
SetTextColor(pdis->hDC, RGB(230,230,230));
HFONT hFontOld = (HFONT) SelectObject(pdis->hDC, font);
DrawText(pdis->hDC, Txtstr, textsize, &pdis->rcItem, DT_SINGLELINE | DT_VCENTER | DT_RIGHT);
SelectObject( pdis->hDC, hFontOld );
}
DeleteDC(hdc);
DeleteBitmap(hCurrIcon);
DeleteBitmap(hIconoff);
DeleteBitmap(hIconon);
font = NULL;
return(RET_OK);
}
答案 0 :(得分:5)
在调用HDC
之前,您需要先将旧对象选回DeleteDC()
。
此外,您似乎没有清理HFONT
返回的CreateFont()
。
答案 1 :(得分:0)
在上面的示例中,您选择了一个对象[〜第21,22行]:
HDC hdc = CreateCompatibleDC(pdis->hDC);
SelectObject(hdc, hCurrIcon); // (*1)
稍后您选择了一个对象
`(if (isText == TRUE))`
HFONT hFontOld = (HFONT) SelectObject(pdis->hDC, font);
DrawText(pdis->hDC, Txtstr, textsize, &pdis->rcItem, DT_SINGLELINE | DT_VCENTER | DT_RIGHT);
并恢复了上一个。对象
SelectObject( pdis->hDC, hFontOld );
但是你从未恢复过第一个SelectObject()
见上面的注释* 1
您之前没有保存以前的(默认)对象。
Lupus Magnus(c)1993
答案 2 :(得分:0)
尝试一下:
在if语句的上方和上方移动hiconoff = ...和hiconon =...。它们将在功能结束时以任何方式删除。修复错误后,您可以找出优化方案。
在第一次运行时,hiconon和hiconoff均未设置为任何值。不要仅仅因为它们是静态数据就假定它们为null。