关于MFC使用GDI绘图的两个容易出错的问题
近期,公司的一个项目是采用华迈SDK来做一套C/S架构的监控中心。自己犯了两个低级错误,写出来看看有多少朋友平时跟我一样没注意,哈哈。
第一个错误是这样出现的:在最初的几个版本中比较顺利。于是有点飘飘然,连续提交了好几个版本都没有进行全面测试。直到一个完成得差不多的版本出现后,来了一次全面的测试。这个时候发现,居然主界面显示不正常。找了好久原因没找到,只好回退到之前最初几个正常的版本,然后建立了个分支专门来排除此问题。一个一个版本的合并,一个一个文件的差异对比,最终定位到一个版本中原来把CXXX::OnPaint()中的CPaintDC dc(this);写成了CClientDC dc(this);
下面是错误分析。
《Windows 程序设计》里曾经说过,在WM_PAINT中一定要成对使用BeginPaint跟EndPaint。否则程序会不停产生WM_PAINT消息。而MFC的CPaintDC的构造与析构中封装了BeginPaint及EndPaint。而CClientDC则没有,所以在OnPaint中使用CClientDC当然是不行的。惭愧啊,惭愧。一时疏忽浪费我整整一个下午找这个BUG。
第二个错误更加惭愧了。直接看码:
void CDxHMContainer::OnPaint()
{
CPaintDC dc(this);
for (UINT i = 0; i < m_vecChilds.size();i++)
{
if (m_vecChilds[i]->IsSelect())
{
CRect rt;
CPen aNewPen,*pOldPen;
int nOldRop = dc.SetROP2(R2_NOTXORPEN);
m_vecChilds[i]->GetClientRect(rt);
rt.InflateRect(2,2);
m_vecChilds[i]->ClientToScreen(rt);
this->ScreenToClient(rt);
aNewPen.CreatePen(PS_SOLID,3,m_colorFrame);
pOldPen = (CPen*)dc.SelectObject(aNewPen);
dc.Rectangle(rt);
dc.SelectObject(pOldPen);// 这里报错
dc.SetROP2(nOldRop);
}
}
}
这段代码很诡异,有时候运行个两三个小时不会出错,有时候三五分钟就出错。请先不要着急看下面的错误分析,你能看出这段代码有什么问题吗?
当初这段代码报错,说实话我一时半会还真没找到原因。最后看MSDN的关于CDC::SelectObject的相关文档,才终于找到原因。
CDC::SelectObject重载了很多种方法。上面代码出错的原因是第一次调用使用的的函数原型是 HGDIOBJ CDC::SelectObject(HGDIOBJ),
而第二次使用的函数原型确是 CBitmap* CDC::SelectObject(CBitmap*)。对于上述代码第一次调用的时候,将HGDIOBJ强行转成CBitmap*实在有点惨不忍睹啊。。。。。。如此,便导致了悲催的崩溃。