pDC->SelectObject返回的指针所指向的GDI对象有可能会被delete掉,所以不能只保存返回的指针。
正确的做法:convert the return pointer to Windows Handle with the GetSafeHdc member function.


For other device contexts, such as those for printers and memory buffers, your assignments can last longer. For these long-life device contexts, things get a little more complicated. The complexity results from the temporary nature of GDI C++ object pointers returned by the SelectObject function. (The temporary "object" will be destroyed by the application framework during the idle loop processing of the application, sometime after the handler function returns the call. See MFC Technical Note #3 in the online documentation.) You can't simply store the pointer in a class data member; instead, you must convert it to a Windows handle (the only permanent GDI identifier) with the GetSafeHdc member function. Here's an example:


// m_pPrintFont points to a CFont object created in CMyView's constructor
// m_hOldFont is a CMyView data member of type HFONT, initialized to 0
void CMyView::SwitchToCourier(CDC* pDC)
{
m_pPrintFont->CreateFont(30, 10, 0, 0, 400, FALSE, FALSE,
0, ANSI_CHARSET, OUT_DEFAULT_PRECIS,
CLIP_DEFAULT_PRECIS, DEFAULT_QUALITY,
DEFAULT_PITCH | FF_MODERN,
"Courier New"); // TrueType
CFont* pOldFont = pDC->SelectObject(m_pPrintFont);
// m_hOldFont is the CGdiObject public data member that stores
//  the handle
m_hOldFont = (HFONT) pOldFont->GetSafeHandle();
}
void CMyView:SwitchToOriginalFont(CDC* pDC)
{
// FromHandle is a static member function that returns an
//  object pointer
if (m_hOldFont) {
pDC->SelectObject(CFont::FromHandle(m_hOldFont));
}
}
// m_pPrintFont is deleted in the CMyView destructor