问题引入:

  最近在尝试编写贪吃蛇游戏时遇到这么一个问题:当系统以较快频率向窗口发送WM_PAINT消息时,调用OnPaint()函数在窗口中绘制图形就会发生闪烁现象。

问题分析:

  当我们把绘图过程放在OnPaint()函数中时(放在OnDraw()函数中也是如此,因为OnDraw()会被OnPaint()调用),由于频繁收到系统的WM_PAINT消息,窗口需要执行重绘。而重绘过程首先是执行了窗口内容的擦除(用当前背景色的画刷对窗口重新绘制),然后再根据绘图语句在窗口客户区中对窗口内容进行重绘。由于频率较快,当前窗口中会产生背景色和窗口内容的反复交替,二者的色差造成了闪烁的效果。

问题解决:

  双缓冲绘图:双缓冲即在内存中创建一个与屏幕绘图区域一致的对象,先将图形绘制到内存中的这个对象上,再一次性将这个对象上的图形拷贝到屏幕上的过程。

  先发上原来OnPaint()函数中的绘图代码:

 void CMainWindow::OnPaint()
{
CClientDC dc(this); CPen White_Pen(PS_SOLID, , RGB(, , ));
CPen *pOldPen = dc.SelectObject(&White_Pen);
CBrush White_Brush(RGB(, , ));
CBrush *pOldBrush = dc.SelectObject(&White_Brush); //Draw background
dc.Rectangle(&m_rcBack); //Draw scoreboard
dc.SelectStockObject(GRAY_BRUSH);
dc.Rectangle(&m_rcScoreBoard); //Draw wall
dc.SelectStockObject(BLACK_BRUSH); int i;
for (i = ; i < MAP_SIZE_CX; ++i)
{
dc.Rectangle(m_rcGameMap[i][]);
dc.Rectangle(m_rcGameMap[i][MAP_SIZE_CY - ]);
}
for (i = ; i < MAP_SIZE_CY - ; ++i)
{
dc.Rectangle(m_rcGameMap[][i]);
dc.Rectangle(m_rcGameMap[MAP_SIZE_CX - ][i]);
} dc.SelectObject(pOldPen);
dc.SelectObject(pOldBrush);
}

  由于出现闪烁的问题将上述代码转写到OnEraseBkgnd()函数中(并且添加了WM_ERASEBKGND的消息映射,顾名思义,就是系统通知窗口擦除客户区的消息),同时在OnPaint()中对OnEraseBkgnd()显式调用(这里要用CPaintDC类,之前用CClientDC类导致WM_PAINT消息在消息队列里造成死循环,原因是因为CClientDC类的构造函数和析构函数里没有像CPaintDC类一样调用::BeginPaint()和::EndPaint()),代码如下:

 void CMainWindow::OnPaint()
{
CPaintDC dc(this);
OnEraseBkgnd(&dc);
}
 BOOL CMainWindow::OnEraseBkgnd(CDC *pDC)
{
CRect rect;
CDC dcMem;
GetClientRect(&rect);;
CBitmap bmp; dcMem.CreateCompatibleDC(pDC);
bmp.CreateCompatibleBitmap(pDC, rect.Width(), rect.Height());
CBitmap *pOldBit = dcMem.SelectObject(&bmp);
dcMem.FillSolidRect(rect, RGB(, , )); //
//Draw
//
CPen White_Pen(PS_SOLID, , RGB(, , ));
CPen *pOldPen = dcMem.SelectObject(&White_Pen);
CBrush White_Brush(RGB(, , ));
CBrush *pOldBrush = dcMem.SelectObject(&White_Brush); //Draw background
dcMem.Rectangle(&m_rcBack); //Draw scoreboard
dcMem.SelectStockObject(GRAY_BRUSH);
dcMem.Rectangle(&m_rcScoreBoard); //Draw wall
dcMem.SelectStockObject(BLACK_BRUSH); int i;
for (i = ; i < MAP_SIZE_CX; ++i)
{
dcMem.Rectangle(m_rcGameMap[i][]);
dcMem.Rectangle(m_rcGameMap[i][MAP_SIZE_CY - ]);
}
for (i = ; i < MAP_SIZE_CY - ; ++i)
{
dcMem.Rectangle(m_rcGameMap[][i]);
dcMem.Rectangle(m_rcGameMap[MAP_SIZE_CX - ][i]);
} pDC->BitBlt(, , rect.Width(), rect.Height(), &dcMem, , , SRCCOPY); dcMem.DeleteDC();
bmp.DeleteObject();
return TRUE;
}

  对比上面两段代码发现,二者的差别仅仅在于前者(OnPaint()函数)是直接对客户区对象(dc)直接进行绘制,而后者(OnEraseBkgnd()函数)是先在内存中开辟了一块基于当前客户区大小的缓冲区(dcMem),用dcMem这一对象取代dc进行相同的绘制操作(二者有相似的成员函数),在内存中绘制完成后再通过如下语句将内存中绘制好的图形直接拷贝到窗口客户区,因为是直接一步完成所以避免了原来的擦除和重绘过程,也就解决了闪烁的问题。

 pDC->BitBlt(, , rect.Width(), rect.Height(), &dcMem, , , SRCCOPY);

总结:

  当窗口客户区需要绘制复杂图形时,如果内存条件允许的话最好采取双缓冲的绘制方法,一方面能解决窗口绘制时闪烁的问题,另一方面,因为减少了窗口反复的擦除过程,也在一定程度上减少了响应时间,是一种用空间换取时间的做法。

参考:

  ①:http://blog.csdn.net/aaahuanian/article/details/7844522

MFC双缓冲绘图(2015.09.24)的更多相关文章

  1. MFC双缓冲绘图实例

    本人之前一直了解双缓冲绘图的基本原理,但是在研究很久之后才大概知道具体的使用过程,本文将详细介绍本人在实际项目中使用双缓冲绘图的案例. 实现功能:主界面显示某张包含人脸的图片,通过dlib detec ...

  2. mfc双缓冲绘图

    1.要求 在界面加载本地图片并显示,每过100ms改变一张图片显示 2.现象 通过定时器控制CImage,Load,Draw,Destroy,会非常的卡顿.因为Load图片时,会是非常大的数据[所有C ...

  3. MFC双缓冲解决图象闪烁[转]

    转载网上找到的一篇双缓冲的文章,很好用.http://www.cnblogs.com/piggger/archive/2009/05/02/1447917.html__________________ ...

  4. 【MFC】MFC绘制动态曲线,用双缓冲绘图技术防闪烁

    摘自:http://zhy1987819.blog.163.com/blog/static/841427882011614103454335/ MFC绘制动态曲线,用双缓冲绘图技术防闪烁   2011 ...

  5. 陈灯WGF双缓冲绘图框架

    “木丸子童屋”,专售各类儿童玩具,价格优惠,请大家多多支持:http://shop65552598.taobao.com/ WGF(windows graphic foundation)为window ...

  6. C#-gdi绘图,双缓冲绘图,Paint事件的触发

    一. 画面闪烁问题与双缓冲技术 1.1 导致画面闪烁的关键原因分析: 1  绘制窗口由于大小位置状态改变进行重绘操作时 绘图窗口内容或大小每改变一次,都要调用Paint事件进行重绘操作,该操作会使画面 ...

  7. 双缓冲绘图和窗口控件的绘制——ATL ActiveX 窗口控件生成向导绘制代码OnDraw的一个错误 .

    双缓冲绘图和窗口控件的绘制 ---ATL ActiveX 窗口控件生成向导绘制代码OnDraw的一个错误 cheungmine 我们通常使用ATL COM组件,生成一个带窗口的ActiveX控件,然后 ...

  8. 简单的 "双缓冲" 绘图的例子(研究一下)

    所谓双缓冲就是先画到内存画布(如: TBitmap), 然后再转帖到目的地. 譬如下面小程序: procedure TForm1.FormCreate(Sender: TObject); begin ...

  9. 简单的 "双缓冲" 绘图的例子

    http://www.cnblogs.com/del/archive/2010/04/19/1715779.html 所谓双缓冲就是先画到内存画布(如: TBitmap), 然后再转帖到目的地. 譬如 ...

随机推荐

  1. 课程笔记:——Javascript 中的预解释1

    1.预解释(变量提升):在当前作用域下,JS代码执行之前,浏览器首先会把所有带var和function关键字的进行提前的声明或者定义var num = 12;声明(declare): var num; ...

  2. AngularJs的UI组件ui-Bootstrap分享(六)——Tabs

    tabs控件使用uib-tabset指令和uib-tab指令,效果是这样的: <!DOCTYPE html> <html ng-app="ui.bootstrap.demo ...

  3. Android Studio 2.2.2 发布

    Android Studio 2.2.2 发布了.包括 canary, dev, beta, and stable 这几个系列. 该版本修复了 Android Studio 2.2.1 中排名较前的b ...

  4. css测试题

    1.CSS 指的是? 您的回答:Computer Style Sheets 正确答案:Cascading Style Sheets 2.在以下的 HTML 中,哪个是正确引用外部样式表的方法? 您的回 ...

  5. jquery实现css3动画

    jquery animate改变元素样式时,只支持数字值的变化,比如width,height等,但是css3属性状态值很多都不是数字值,而是字符串和数字混合在一起,比如translate(), rot ...

  6. 可滑动的ExpandableListView

    可以向左滑动的扩展列表 向左滑动源码是参照GitHub上的里的 ListView的思路写出来的,按照他的思路,由于本人水平有限,只写了关键代码,能够完美运行,adapter改变之后能自动收回. 滑出状 ...

  7. C语言程序设计第二次作业

    一.学习内容 掌握关系运算符.逻辑运算符.条件运算符 掌握常用数学函数的用法 if语句(单分支,双分支和多分支) 用switch语句实现多分支 理解多个if语句,if...else if... 和if ...

  8. 将IIS6.0中的某页设置成首页

  9. C#中datatable导出excel(三种方法)

    方法一:(拷贝直接可以使用,适合大批量资料, 上万笔) Microsoft.Office.Interop.Excel.Application appexcel = new Microsoft.Offi ...

  10. A与B相交后的图形查询

    按照A与B图形得到相交后的图斑 <!-- <%@ Page Language="C#" AutoEventWireup="true" CodeBeh ...