关于VC画图闪屏的问题 不闪屏的条件: a. 你不能让 WM_ERASEBACKGROUND 动作 b. WM_PAINT 时, 要有内存镜象直接BitBlt也就是视觉上不存在顿时的画面清除而所有 Windows Common Control 都没有做到这两点 解决方案 » 免费领取超大流量手机卡,每月29元包185G流量+100分钟通话, 中国电信官方发货 在WM_PAINT响应函数下使用创建兼容DC的方法重画画面,你无法彻底解决这个问题,除非采用DirectX直接写屏. 这是一个gui编程常出现的问题,当你的绘图过程时间比较长的时候,gui总是会出现闪烁,最直接的办法当然是用directx如果你不想用directx的话,你可以在ondraw()函数中(或者响应wm_paint的代码中),先调用GetUpdateRgn,获得需要重画的区域,然后你只需要更新该区域对应的图像区域(获得对应的图像区域,你可以看看图形学中相关的理论),这样的话就大大降低了你绘图的量,闪烁也就减少了。不至于要我给你源代码吧,呵呵 同意gancheng:你可以在激发ApdateAllviews(kkk,iii,lll)中给出重画区域的rect参数...然后在OnUpdate()中将此区域InvalidateRect(&rect)...最后在OnDraw()中截割rect...仅仅重画这个rect中的内容...这样做效果会好一点,如果对图形效果要求比较高(比如做游戏软件)就用directx吧!! 我正在做这个,也发现友闪屏的现象,我现在使用的是BITBLT();今天我看了处理这个问题有两个方法: 都是使用BITBLT(); 1: 使用内存镜像:创建一个内存区域,创建位图--->画图--->BITBLT() 2: 创建两个屏幕区域大小->>>>在上画图,---判断边界问题--->BITBLT() 摘自:电脑编程技巧与维护 1999---2001(没注意那一期的) 错,无论使用不使用memDC,如果让WM_ERASEBACKGROUND 自动工作,则肯定会闪。因此必须重载WM_ERASEBACKGROUND消息,不允许系统自动处理这个消息,强行返回true就可以了 代码:void CGraphView::OnDraw(CDC* pDC){ CRect rect; GetClientRect( &rect ); CDC dcMem; dcMem.CreateCompatibleDC( pDC ); CBitmap bmp; bmp.CreateCompatibleBitmap( pDC, rect.right, rect.bottom ); CBrush brush; brush.CreateSolidBrush( RGB( 255, 255, 255 ) ); dcMem.SelectObject( &bmp ); dcMem.FillRect( &rect, &brush ); CGraphDoc* pDoc = GetDocument(); ASSERT_VALID(pDoc); // 加上drawing代码,注意是对dcMem进行操作 pDC->BitBlt( 0, 0, rect.right, rect.bottom, &dcMem, 0, 0, SRCCOPY );}原理就是创建一个兼容的内存DC,之后select一个与client同大小的bitmap进去,以后就可以直接对内存DC绘图了,绘图完毕后再BitBlt到要draw的dc 对了,补充一句,调用InvalidateRect()时,不要erase background,不然任何防闪烁的代码都是白搭。只要做到我说的,保证没有闪烁。就是速度略慢点。调用GetUpdateRgn()的方法并不一定能保证没有闪烁,没有闪烁,关键就是要改变颜色的点只能改变一次颜色,即只重画一次,如果重画多了就会有闪烁。所以一次性bitblt到dc不会有闪烁。 用MemDC肯丁行,你用的有问题 这个问题我也问过,可目前还没解决,我用了ATField(field)所说的方法,仍然有问题,跟原来直接在ondraw里直接画的唯一区别就是成块成块的闪了。 有2个简单的方法:1. Invalidate(FALSE);2. 计算出要重画的区域,形成一个RECT,然后调用InvalidateRect(&rect); 我也遇到这个问题,但我是想获得滚动条的指针,请高手赐教,我想通过控制滚动条来操纵视图!!如有相关的例子请给我发到[email protected] 内存dc能够解决;其实,闪烁是由于你在一次更新屏幕时,向屏幕上画了多次;使用内存dc,就可以解决;你一次更新,只需要向屏幕画一次;其他的在内存中处理; MFC如何高效地绘图[ 作者: TouchMe 添加时间: 2001-12-25 8:21:34 ] 显示图形如何避免闪烁,如何提高显示效率是问得比较多的问题。而且多数人认为MFC的绘图函数效率很低,总是想寻求其它的解决方案。MFC的绘图效率的确不高但也不差,而且它的绘图函数使用非常简单,只要使用方法得当,再加上一些技巧,用MFC可以得到效率很高的绘图程序。我想就我长期(呵呵当然也只有2年多)使用MFC绘图的经验谈谈我的一些观点。1、显示的图形为什么会闪烁? 我们的绘图过程大多放在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、如何避免闪烁 在知道图形显示闪烁的原因之后,对症下药就好办了。首先当然是去掉MFC提供的背景绘制过程了。实现的方法很多, * 可以在窗口形成时给窗口的注册类的背景刷付NULL * 也可以在形成以后修改背景static CBrush brush(RGB(255,0,0));SetClassLong(this->m_hWnd,GCL_HBRBACKGROUND,(LONG)(HBRUSH)brush); * 要简单也可以重载OnEraseBkgnd(CDC* pDC)直接返回TRUE 这样背景没有了,结果图形显示的确不闪了,但是显示也象前面所说的一样,变得一团乱。怎么办?这就要用到双缓存的方法了。双缓冲就是除了在屏幕上有图形进行显示以外,在内存中也有图形在绘制。我们可以把要显示的图形先在内存中绘制好,然后再一次性的将内存中的图形按照一个点一个点地覆盖到屏幕上去(这个过程非常快,因为是非常规整的内存拷贝)。这样在内存中绘图时,随便用什么反差大的背景色进行清除都不会闪,因为看不见。当贴到屏幕上时,因为内存中最终的图形与屏幕显示图形差别很小(如果没有运动,当然就没有差别),这样看起来就不会闪。3、如何实现双缓冲 首先给出实现的程序,然后再解释,同样是在OnDraw(CDC *pDC)中:CDC MemDC; //首先定义一个显示设备对象CBitmap MemBitmap;//定义一个位图对象//随后建立与屏幕显示兼容的内存显示设备MemDC.CreateCompatibleDC(NULL);//这时还不能绘图,因为没有地方画 ^_^//下面建立一个与屏幕显示兼容的位图,至于位图的大小嘛,可以用窗口的大小MemBitmap.CreateCompatibleBitmap(pDC,nWidth,nHeight); //将位图选入到内存显示设备中//只有选入了位图的内存显示设备才有地方绘图,画到指定的位图上CBitmap *pOldBit=MemDC.SelectObject(&MemBitmap);//先用背景色将位图清除干净,这里我用的是白色作为背景//你也可以用自己应该用的颜色MemDC.FillSolidRect(0,0,nWidth,nHeight,RGB(255,255,255));//绘图MemDC.MoveTo(……);MemDC.LineTo(……);//将内存中的图拷贝到屏幕上进行显示pDC->BitBlt(0,0,nWidth,nHeight,&MemDC,0,0,SRCCOPY);//绘图完成后的清理MemBitmap.DeleteObject();MemDC.DeleteDC();上面的注释应该很详尽了,废话就不多说了。4、如何提高绘图的效率 我主要做的是电力系统的网络图形的CAD软件,在一个窗口中往往要显示成千上万个电力元件,而每个元件又是由点、线、圆等基本图形构成。如果真要在一次重绘过程重画这么多元件,可想而知这个过程是非常漫长的。如果加上了图形的浏览功能,鼠标拖动图形滚动时需要进行大量的重绘,速度会慢得让用户将无法忍受。怎么办?只有再研究研究MFC的绘图过程了。 实际上,在OnDraw(CDC *pDC)中绘制的图并不是所有都显示了的,例如:你在OnDraw中画了两个矩形,在一次重绘中虽然两个矩形的绘制函数都有执行,但是很有可能只有一个显示了,这是因为MFC本身为了提高重绘的效率设置了裁剪区。裁剪区的作用就是:只有在这个区内的绘图过程才会真正有效,在区外的是无效的,即使在区外执行了绘图函数也是不会显示的。因为多数情况下窗口重绘的产生大多是因为窗口部分被遮挡或者窗口有滚动发生,改变的区域并不是整个图形而只有一小部分,这一部分需要改变的就是pDC中的裁剪区了。因为显示(往内存或者显存都叫显示)比绘图过程的计算要费时得多,有了裁剪区后显示的就只是应该显示的部分,大大提高了显示效率。但是这个裁剪区是MFC设置的,它已经为我们提高了显示效率,在进行复杂图形的绘制时如何进一步提高效率呢?那就只有去掉在裁剪区外的绘图过程了。可以先用pDC->GetClipBox()得到裁剪区,然后在绘图时判断你的图形是否在这个区内,如果在就画,不在就不画。如果你的绘图过程不复杂,这样做可能对你的绘图效率不会有提高。 你的程序中用了我的方法后仍然闪烁的原因因该是你没有防止意外的erase,把代码贴出来吧。 如何响应子对话框的控件消息 extern 多变量用法? 一个类似 "广播信息 "的网络程序问题 怎么设置菜单的快捷方式如F2为修改 而不用去点菜单 串口问题,第一次启动无法监控串口信息,使用一次超级终端连接就可以了,请问是什么问题? 求购VC进销存源码 如何 让windows知道 *.abc 文件 由 ABC.exe 打开? 如何判断两个绘制的文字是否重合呢? 在vc中如何统计代码行数,谢谢! CFileDialog对话框能选定多个文件吗? 高分求救,我的xml练习程序狂吃内存!!!!! 新手上路:请问谁用过CTreeCtrl::SortChildrenCB,能帮我讲讲或者有代码吗?
如果你不想用directx的话,你可以在ondraw()函数中(或者响应wm_paint的代码中),先调用GetUpdateRgn,获得需要重画的区域,然后你只需要更新该区域对应的图像区域(获得对应的图像区域,你可以看看图形学中相关的理论),这样的话就大大降低了你绘图的量,闪烁也就减少了。
不至于要我给你源代码吧,呵呵
你可以在激发ApdateAllviews(kkk,iii,lll)中给出重画区域的rect参数...
然后在OnUpdate()中将此区域InvalidateRect(&rect)...
最后在OnDraw()中截割rect...仅仅重画这个rect中的内容...
这样做效果会好一点,如果对图形效果要求比较高(比如做游戏软件)就用directx吧!!
都是使用BITBLT();
1:
使用内存镜像:创建一个内存区域,创建位图--->画图--->BITBLT()
2:
创建两个屏幕区域大小->>>>在上画图,---判断边界问题--->BITBLT()
摘自:电脑编程技巧与维护 1999---2001(没注意那一期的)
则肯定会闪。因此必须重载WM_ERASEBACKGROUND消息,不允许系统
自动处理这个消息,强行返回true就可以了
void CGraphView::OnDraw(CDC* pDC)
{
CRect rect;
GetClientRect( &rect );
CDC dcMem;
dcMem.CreateCompatibleDC( pDC );
CBitmap bmp;
bmp.CreateCompatibleBitmap( pDC, rect.right, rect.bottom );
CBrush brush;
brush.CreateSolidBrush( RGB( 255, 255, 255 ) );
dcMem.SelectObject( &bmp );
dcMem.FillRect( &rect, &brush );
CGraphDoc* pDoc = GetDocument();
ASSERT_VALID(pDoc); // 加上drawing代码,注意是对dcMem进行操作 pDC->BitBlt( 0, 0, rect.right, rect.bottom, &dcMem, 0, 0, SRCCOPY );
}
原理就是创建一个兼容的内存DC,之后select一个与client同大小的bitmap
进去,以后就可以直接对内存DC绘图了,绘图完毕后再BitBlt到要draw的dc
只要做到我说的,保证没有闪烁。就是速度略慢点。
调用GetUpdateRgn()的方法并不一定能保证没有闪烁,
没有闪烁,关键就是要改变颜色的点只能改变一次颜色,即只重画一次,如果重画多了就会有闪烁。所以一次性bitblt到dc不会有闪烁。
1. Invalidate(FALSE);
2. 计算出要重画的区域,形成一个RECT,然后调用InvalidateRect(&rect);
其实,闪烁是由于你在一次更新屏幕时,向屏幕上画了多次;
使用内存dc,就可以解决;你一次更新,只需要向屏幕画一次;
其他的在内存中处理;
而且多数人认为MFC的绘图函数效率很低,总是想寻求其它的解决方案。
MFC的绘图效率的确不高但也不差,而且它的绘图函数使用非常简单,
只要使用方法得当,再加上一些技巧,用MFC可以得到效率很高的绘图程序。
我想就我长期(呵呵当然也只有2年多)使用MFC绘图的经验谈谈
我的一些观点。1、显示的图形为什么会闪烁?
我们的绘图过程大多放在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、如何避免闪烁
在知道图形显示闪烁的原因之后,对症下药就好办了。首先当然是去掉MFC
提供的背景绘制过程了。实现的方法很多,
* 可以在窗口形成时给窗口的注册类的背景刷付NULL
* 也可以在形成以后修改背景
static CBrush brush(RGB(255,0,0));
SetClassLong(this->m_hWnd,GCL_HBRBACKGROUND,(LONG)(HBRUSH)brush);
* 要简单也可以重载OnEraseBkgnd(CDC* pDC)直接返回TRUE
这样背景没有了,结果图形显示的确不闪了,但是显示也象前面所说的一样,
变得一团乱。怎么办?这就要用到双缓存的方法了。双缓冲就是除了在屏幕上有
图形进行显示以外,在内存中也有图形在绘制。我们可以把要显示的图形先在内存中
绘制好,然后再一次性的将内存中的图形按照一个点一个点地覆盖到屏幕上去(这个
过程非常快,因为是非常规整的内存拷贝)。这样在内存中绘图时,随便用什么反差
大的背景色进行清除都不会闪,因为看不见。当贴到屏幕上时,因为内存中最终的图形
与屏幕显示图形差别很小(如果没有运动,当然就没有差别),这样看起来就不会闪。
3、如何实现双缓冲
首先给出实现的程序,然后再解释,同样是在OnDraw(CDC *pDC)中:CDC MemDC; //首先定义一个显示设备对象
CBitmap MemBitmap;//定义一个位图对象//随后建立与屏幕显示兼容的内存显示设备
MemDC.CreateCompatibleDC(NULL);
//这时还不能绘图,因为没有地方画 ^_^
//下面建立一个与屏幕显示兼容的位图,至于位图的大小嘛,可以用窗口的大小
MemBitmap.CreateCompatibleBitmap(pDC,nWidth,nHeight);
//将位图选入到内存显示设备中
//只有选入了位图的内存显示设备才有地方绘图,画到指定的位图上
CBitmap *pOldBit=MemDC.SelectObject(&MemBitmap);//先用背景色将位图清除干净,这里我用的是白色作为背景
//你也可以用自己应该用的颜色
MemDC.FillSolidRect(0,0,nWidth,nHeight,RGB(255,255,255));//绘图
MemDC.MoveTo(……);
MemDC.LineTo(……);//将内存中的图拷贝到屏幕上进行显示
pDC->BitBlt(0,0,nWidth,nHeight,&MemDC,0,0,SRCCOPY);//绘图完成后的清理
MemBitmap.DeleteObject();
MemDC.DeleteDC();上面的注释应该很详尽了,废话就不多说了。
4、如何提高绘图的效率
我主要做的是电力系统的网络图形的CAD软件,在一个窗口中往往要显示成千上万个电力元件,而每个元件又是由点、线、圆等基本图形构成。如果真要在一次重绘过程重画这么多元件,可想而知这个过程是非常漫长的。如果加上了图形的浏览功能,鼠标拖动图形滚动时需要进行大量的重绘,速度会慢得让用户将无法忍受。怎么办?只有再研究研究MFC的绘图过程了。
实际上,在OnDraw(CDC *pDC)中绘制的图并不是所有都显示了的,例如:你
在OnDraw中画了两个矩形,在一次重绘中虽然两个矩形的绘制函数都有执行,但是很有可能只有一个显示了,这是因为MFC本身为了提高重绘的效率设置了裁剪区。裁剪区的作用就是:只有在这个区内的绘图过程才会真正有效,在区外的是无效的,即使在区外执行了绘图函数也是不会显示的。因为多数情况下窗口重绘的产生大多是因为窗口部分被遮挡或者窗口有滚动发生,改变的区域并不是整个图形而只有一小部分,这一部分需要改变的就是pDC中的裁剪区了。因为显示(往内存或者显存都叫显示)比绘图过程的计算要费时得多,有了裁剪区后显示的就只是应该显示的部分,大大提高了显示效率。但是这个裁剪区是MFC设置的,它已经为我们提高了显示效率,在进行复杂图形的绘制时如何进一步提高效率呢?那就只有去掉在裁剪区外的绘图过程了。可以先用pDC->GetClipBox()得到裁剪区,然后在绘图时判断你的图形是否在这个区内,如果在就画,不在就不画。
如果你的绘图过程不复杂,这样做可能对你的绘图效率不会有提高。
把代码贴出来吧。