void CStringView::OnDraw(CDC* pDC)
{
int i,j,nHeight;
CString str;
CFont font;
TEXTMETRIC tm; CPoemDoc* pDoc = GetDocument();
ASSERT_VALID(pDoc); pDC->Rectangle(m_rectPrint+CRect(0,0,-20,20));
j=m_rectPrint.Width()/1440;
for(i=0;i<=j;i++){
str.Format("%02d",i);
pDC->TextOut(i*1440,0,str);
}
j=-m_rectPrint.Height()/1440;
for(i=0;i<=j;i++){
str.Format("%02d",i);
pDC->TextOut(0,-i*1440,str);
}
font.CreateFont(-400,0,0,0,400,false,false,0,ANSI_CHARSET,
OUT_DEFAULT_PRECIS,CLIP_DEFAULT_PRECIS,
DEFAULT_QUALITY,DEFAULT_PITCH|FF_ROMAN,
"Times New Roman");
CFont* pOldFont=(CFont*)pDC->SelectObject(&font);
pDC->GetTextMetrics(&tm);
nHeight=tm.tmHeight+tm.tmExternalLeading; j=pDoc->m_stringArray.GetSize();
for(i=0;i<j;i++){
pDC->TextOut(720,-i*nHeight-720,pDoc->m_stringArray[i]);
} pDC->SelectObject(pOldFont);

}
上面程序中为什么要加入最后一行pDC->SelectObject(pOldFont);呢?我试过不加入也可以正常运行

解决方案 »

  1.   

    因为传进来上个CDC* pDC 的默认Font对象,当结束这个OnDraw调用后 CDC将根据这个CFont句柄释放这个CFont对象,否则OldFont对象的指针丢失,无法找到这个对象,内存泄漏而最常用的做法:因为CFont 一般使用的是局部成员,当SelectObject使用完这个对象的时候需要调用DeleteObject删除这个CFont对象,当然删除这个对象的时候你必须使CDC断开这个对象的句柄,才能安全删除,避免出现野指针情情...
      

  2.   

    恢复CDC,释放新创建的GDI对象CFont,否则会有GDI资源泄漏,
    CFont font;
    你是申明了一个局部GDI对象,出了作用域是会自动释放,
    但良好的编程习惯要: 有Create 就要有DeleteObject相对应(手工释放GDI对象,也就是pDC->SelectObject(pOldFont);后要加一句:font.DeleteObject() )。
      

  3.   

    记得一定要恢复,否则 pOldFont 之类GDI对象每运行一次它所在的 "函数" 就会增加一个泄漏,是很严重的问题。