显示的图形为什么会闪烁
我们的绘图过程大多放在OnDraw或者OnPaint函数中,OnDraw在进行屏幕显示时是由OnPaint进行调用的。当窗口由于任何原因需要重绘时,总是先用背景色将显示区清除,然后才调用OnPaint,而背景色往往与绘图内容反差很大,这样在短时间内背景色与显示图形的交替出现,使得显示窗口看起来在闪。如果将背景刷设置成NULL,这样无论怎样重绘图形都不会闪了。当然,这样做会使得窗口的显示乱成一团,因为重绘时没有背景色对原来绘制的图形进行清除,而又叠加上了新的图形。有的人会说,闪烁是因为绘图的速度太慢或者显示的图形太复杂造成的,其实这样说并不对,绘图的显示速度对闪烁的影响不是根本性的。例如在OnDraw(CDC *pDC)中这样写:
pDC->MoveTo(0,0);
pDC->LineTo(100,100);
这个绘图过程应该是非常简单、非常快了吧,但是拉动窗口变化时还是会看见闪烁。其实从道理上讲,画图的过程越复杂越慢闪烁应该越少,因为绘图用的时间与用背景清除屏幕所花的时间的比例越大人对闪烁的感觉会越不明显。比如:清楚屏幕时间为1s绘图时间也是为1s,这样在10s内的连续重画中就要闪烁5次;如果清楚屏幕时间为1s不变,而绘图时间为9s,这样10s内的连续重画只会闪烁一次。这个也可以试验,在OnDraw(CDC *pDC)中这样写:
for(int i=0;i<100000;i++)
{
pDC->MoveTo(0,i);
pDC->LineTo(1000,i);
}
程序有点极端,但是能说明问题。
说到这里可能又有人要说了,为什么一个简单图形看起来没有复杂图形那么闪呢?这是因为复杂图形占的面积大,重画时造成的反差比较大,所以感觉上要闪得厉害一些,但是闪烁频率要低。那为什么动画的重画频率高,而看起来却不闪?这里,我就要再次强调了,闪烁是什么?闪烁就是反差,反差越大,闪烁越厉害。因为动画的连续两个帧之间的差异很小所以看起来不闪。如果不信,可以在动画的每一帧中间加一张纯白的帧,不闪才怪呢。
2、解决办法:
在图形图象处理编程过程中,双缓冲是一种基本的技术。我们知道,如果窗体在响应WM_PAINT消息的时候要进行复杂的图形处理,那么窗体在重绘时由于过频的刷新而引起闪烁现象。解决这一问题的有效方法就是双缓冲技术。
因为窗体在刷新时,总要有一个擦除原来图象的过程OnEraseBkgnd,它利用背景色填充窗体绘图区,然后在调用新的绘图代码进行重绘,这样一擦一写造成了图象颜色的反差。当WM_PAINT的响应很频繁的时候,这种反差也就越发明显。于是我们就看到了闪烁现象。
我们会很自然的想到,避免背景色的填充是最直接的办法。但是那样的话,窗体上会变的一团糟。因为每次绘制图象的时候都没有将原来的图象清除,造成了图象的残留,于是窗体重绘时,画面往往会变的乱七八糟。所以单纯的禁止背景重绘是不够的。我们还要进行重新绘图,但要求速度很快,于是我们想到了使用BitBlt函数。它可以支持图形块的复制,速度很快。我们可以先在内存中作图,然后用此函数将做好的图复制到前台,同时禁止背景刷新,这样就消除了闪烁。以上也就是双缓冲绘图的基本的思路。
3、具体步骤:
假设我们建立了一个Draw的工程,我们要在DrawView中进行绘图操作。
在双缓冲方法中,首先要做的是屏蔽背景刷新。背景刷新其实是在响应WM_ERASEBKGND消息。我们在视类(CDrawView)中添加对这个消息的响应,可以看到缺省的代码如下:
BOOL CDrawView::OnEraseBkgnd(CDC* pDC)
{
//return CDrawView::OnEraseBkgnd(pDC);
return TRUE;
}
接下来是双缓冲的实现步骤:
(1)增加成员变量(在DrawView.h文件中)
//参数声明
CBitmap* m_pOldBitmap;
CBitmap* m_pMemBitmap; //声明内存中承载临时图象的位图
CDC* m_pMemDC; //声明用于缓冲作图的内存DC
(2)初始化变量(在DrawView的构造函数中)
m_pMemDC=new CDC();
m_pMemBitmap=new CBitmap();
(3)增加消息响应函数WM_CREATE(在DrawView.cpp中)
int CDrawView::OnCreate(LPCREATESTRUCT lpCreateStruct)
{
if (CView::OnCreate(lpCreateStruct) == -1)
return -1;
int x=GetSystemMetrics(SM_CXSCREEN);
int y=GetSystemMetrics(SM_CYSCREEN);
CDC* pDC=GetDC();
m_pMemDC->CreateCompatibleDC(pDC); //依附窗口DC创建兼容内存DC
m_pMemBitmap->CreateCompatibleBitmap(pDC,x,y); //创建兼容位图
m_pOldBitmap=m_pMemDC->SelectObject(m_pMemBitmap); //将位图选进内存DC,原位图保存到m_pOldBitmap
CBrush brush(RGB(255,255,255));
m_pMemDC->FillRect(CRect(0,0,x,y),&brush); //设置客户区背景为白色
ReleaseDC(pDC);
return 0;
}
(4)修改OnDraw()函数(DrawView.cpp中)
void CDrawView::OnDraw(CDC* pDC)
{
CDocument* pDoc = GetDocument();
CRect rc;
GetClientRect(&rc);
DrawSomething(); //在这个函数里你可以画你想画的东西
pDC->BitBlt(0,0,rc.Width(),rc.Height(),m_pMemDC,0,0,SRCCOPY);
//这里就是将内存里面的画布复制到显示设备的buffer了
}
(5)自己的绘图函数(DrawView.cpp中)
void DrawSomething()
{
m_pMemDC->Rectangle(0,0,100,100); //此处画了个矩形
Invalidate();
}
(6)delete掉new的东西(在DrawView的析构函数中)
delete m_pBitmap;
delete m_pMemDC;