双缓冲法解决重绘和闪屏问题
重绘导致原因:UpdateData、Invalidate、InvalidateRect和UpdateWindow函数。
1. UpdateData重绘控件函数
UpdateData(TRUE)——刷新控件的值到对应的变量。(外部输入值交给内部变量)
UpdateData(FALSE) —— 拷贝变量值到控件显示。(变量的最终运算结果值交给外部输出显示)
即:变量值—>控件显示。
2. Invalidate()
该函数的作用是使整个窗口客户区无效。窗口的客户区无效意味着需要重绘,例如,如果一个被其它窗口遮住的窗口变成了前台窗口,那么原来被遮住的部分就是无效的,需要重绘。这时Windows会在应用程序的消息队列中放置WM_PAINT消息。MFC为窗口类提供了WM_PAINT的消息处理函数OnPaint,OnPaint负责重绘窗口。视图类有一些例外,在视图类的OnPaint函数中调用了OnDraw函数,实际的重绘工作由OnDraw来完成。参数bErase为TRUE时,重绘区域内的背景将被擦除,否则,背景将保持不变。
3. InvalidateRect
用InvalidateRect函数只重绘部分区域,而且不重绘背景(第二个参数用FALSE)就可以解决大部分的屏闪问题。
比如:CRect rect(10,47,10+120,47+70);
InvalidateRect(rect,FALSE);
4. UpdateWindow函数
UpdateWindow( )的作用是使窗口立即重绘。调用Invalidate等函数后窗口不会立即重绘,这是由于WM_PAINT消息的优先级很低,它需要等消息队列中的其它消息发送完后才能被处理。调用UpdateWindow函数可使WM_PAINT被直接发送到目标窗口,从而导致窗口立即重绘。注意:函数绕过应用程序的消息队列,直接发送WM_PAINT消息给指定窗口的窗口过程,如果更新区域为空,则不发送消息。
解决方法:
双缓冲是一种基本的技术。我们知道,如果窗体在响应WM_PAINT消息的时候要进行复杂的图形处理,那么窗体在重绘时由于过频的刷新而引起闪烁现象。解决这一问题的有效方法就是双缓冲技术。
因为窗体在刷新时,总要有一个擦除原来图象的过程,它利用背景色填充窗体绘图区,然后在调用新的绘图代码进行重绘,这样一擦一写造成了图象颜色的反差。当WM_PAINT的响应很频繁的时候,这种反差也就越发明显。于是我们就看到了闪烁现象。(
当窗口由于任何原因需要重绘时,
总是先用背景色将显示区清除,然后才调用OnPaint,而背景色往往与绘图内容
反差很大,这样在短时间内背景色与显示图形的交替出现,使得显示窗口看起来
在闪。如果将背景刷设置成NULL,这样无论怎样重绘图形都不会闪了。
当然,这样做会使得窗口的显示乱成一团,因为重绘时没有背景色对原来
绘制的图形进行清除,而又叠加上了新的图形。) 我们会很自然的想到,避免背景色的填充是最直接的办法。但是那样的话,窗体上会变的一团糟。因为每次绘制图象的时候都没有将原来的图象清除,造成了图象的残留,于是窗体重绘时,画面往往会变的乱七八糟。所以单纯的禁止背景重绘是不够的。我们还要进行重新绘图,但要求速度很快,于是我们想到了使用BitBlt函数。它可以支持图形块的复制,速度很快。我们可以先在内存中作图,然后用此函数将做好的图复制到前台,同时禁止背景刷新,这样就消除了闪烁。以上也就是双缓冲绘图的基本的思路。
BitBlt函数执行颜色数据的位块传送,从指定的源设备描述表向给定的一个目的设备描述表传送对应于一个象素矩形的颜色数据。
BOOL BitBlt(HDC hdcDest,
int nXDest,
int nYDest,
int nWidth,
int nHeight,
HDC hdcSrc,
int nXSrc,
int nYSrc,
DWORD dwRop);
具体实现:在窗口类中定义成员变量与成员函数:
CBrush m_brush;CDC m_memDC;//画在内存上CBitmap m_Bmp;CWnd *m_pDrawWnd;void InitialDBB();void DrawOnMem(double *pdData, unsigned long DataLen);void DrawOnStaticArea(double *pdData, unsigned long DataLen);在该类的初始化函数中添加BOOL CdaexpDlg::OnInitDialog(){
。。。。。。
m_brush.CreateSolidBrush(RGB(25,200,25));m_pDrawWnd=GetDlgItem(IDC_PICTURE);InitialDBB();
。。。。。。
}在重绘函数中添加void CdaexpDlg::OnPaint()
{
。。。
//已经有图像保存在缓冲区了PAINTSTRUCT ps;CRect rt;m_pDrawWnd->GetClientRect(&rt); CDC* pDC=m_pDrawWnd->BeginPaint(&ps);pDC->BitBlt(0, 0, rt.Width(), rt.Height(), &m_memDC, 0, 0, SRCCOPY); m_pDrawWnd->EndPaint(&ps);。。。
}/******************************************************************************************************初始化内存和画布
*******************************************************************************************************/void CdaexpDlg::InitialDBB()
{CRect rt;m_pDrawWnd->GetClientRect(&rt);CDC* sDC = m_pDrawWnd->GetDC();// 为屏幕DC创建兼容的内存DCif(!m_memDC.CreateCompatibleDC(sDC))//{ ::PostQuitMessage(0);} m_nHeight = rt.bottom - rt.top;m_nWidth = rt.right - rt.left;// 创建位图,不能是m_memDC,否则无颜色m_Bmp.CreateCompatibleBitmap(sDC, rt.Width(), rt.Height());//m_memDC// 相当于选择画布,m_pDrawWnd->::SelectObject(m_memDC.GetSafeHdc(), m_Bmp); m_pDrawWnd->ReleaseDC(sDC);
}/******************************************************************************************************绘制输出波形至内存
*******************************************************************************************************/
void CdaexpDlg::DrawOnMem( )
{unsigned long i;
//---------------------------------------------------//获取绘图区域CRect rect;m_pDrawWnd->GetClientRect(&rect);//----------------------------------------------------//将背景涂黑COLORREF crl = RGB(0,0,0);m_memDC.FillSolidRect(rect, crl);
//----------------------------------------------------//设置画笔为黑色CPen pen(PS_SOLID,1,RGB(25,200,25));m_memDC.SelectObject(&pen);// ======================================================// = 画出波形m_memDC.MoveTo( , );m_memDC.LineTo( , );}/******************************************************************************************************将内存内的波形绘制到界面
*******************************************************************************************************/
void CdaexpDlg::DrawOnStaticArea( )
{CWnd* pWnd = GetDlgItem(IDC_PICTURE);//获得静态文本框的窗口对象CRect rect;pWnd->GetClientRect(&rect); CDC* pDC = pWnd->GetDC();DrawOnMem( );// 一次性的将内存设备环境上绘制完毕的图形"贴"到屏幕上pDC->BitBlt(0, 0, rect.Width(), rect.Height(), &m_memDC, 0, 0, SRCCOPY);pWnd->ReleaseDC(pDC);
}
如何提高绘图的效率
电力系统的网络图形的CAD软件,在一个窗口中往往要显示成千上万个电力元件,而每个元件又是由点、线、圆等基本图形构成。如果真要在一次重绘过程重画这么多元件,可想而知这个过程是非常漫长的。如果加上了图形的浏览功能,鼠标拖动图形滚动时需要进行大量的重绘,速度会慢得让用户将无法忍受。怎么办?只有再研究研究MFC的绘图过程了。
实际上,在OnDraw(CDC *pDC)中绘制的图并不是所有都显示了的,例如:你
在OnDraw中画了两个矩形,在一次重绘中虽然两个矩形的绘制函数都有执行,但是很有可能只有一个显示了,这是因为MFC本身为了提高重绘的效率设置了裁剪区。裁剪区的作用就是:只有在这个区内的绘图过程才会真正有效,在区外的是无效的,即使在区外执行了绘图函数也是不会显示的。因为多数情况下窗口重绘的产生大多是因为窗口部分被遮挡或者窗口有滚动发生,改变的区域并不是整个图形而只有一小部分,这一部分需要改变的就是pDC中的裁剪区了。因为显示(往内存或者显存都叫显示)比绘图过程的计算要费时得多,有了裁剪区后显示的就只是应该显示的部分,大大提高了显示效率。但是这个裁剪区是MFC设置的,它已经为我们提高了显示效率,在进行复杂图形的绘制时如何进一步提高效率呢?那就只有去掉在裁剪区外的绘图过程了。可以先用pDC->GetClipBox()得到裁剪区,然后在绘图时判断你的图形是否在这个区内,如果在就画,不在就不画。
如果你的绘图过程不复杂,这样做可能对你的绘图效率不会有提高。
双缓冲法解决重绘和闪屏问题相关推荐
- winform上控件太多,绘制时会逐个出现,通常说双缓冲能解决但实际不能解决的问题的解决方法。
winform上控件太多,绘制时会逐个出现,通常说双缓冲能解决但实际不能解决的问题的解决方法. 参考文章: (1)winform上控件太多,绘制时会逐个出现,通常说双缓冲能解决但实际不能解决的问题的解 ...
- UITableviewcell重用机制以及解决重绘出现的重叠现象
2019独角兽企业重金招聘Python工程师标准>>> UITableviewcell重用机制以及解决重绘出现的重叠现象 重点1.删除子控件 重点2.重新分配控件 该方法是自定义创建 ...
- recycler 刷新图片闪烁_android 解决RecyclerView notifyDataSetChanged刷新闪屏问题(图片刷新)...
最近遇到了RecyclerView 在执行notifyDataSetChanged操作时会出现闪屏,图片刷新,然后根据网上资料提示使用 修改动画,设置动画时间 recyclerView.getIte ...
- 解决Chrome播放视频闪屏黑屏无法播放
解决方法 先说解决方法:Chrome→设置→硬件加速→关闭.如下图. 之前遇到的问题 使用 Chrome 浏览网页的时候,遇到自动播放的视频,有大概率会整个 Chrome 浏览器闪一下,闪屏变黑一秒钟 ...
- Android一种实现夜间模式方式,同时解决调用recreate() 时闪屏问题
最近项目中要实现夜间模式,调研了几种Android的夜间模式实现方式,都是需要动态的改变ui界面引用到color.drawable等资源文件.由于这个项目是迭代了比较多版本的,ui界面比较多,项目也比 ...
- win10间歇性闪屏_如何解决Win10系统频繁闪屏刷新问题?
Win10系统频繁闪屏刷新该如何解决?在使用Win10系统过程中难免会碰到一些情况,比如有用户发现电脑会出现频繁闪屏刷新,看着非常不舒服,该怎么办呢?电脑闪屏,如果确定不是硬件问题的话,那么就有可能是 ...
- 解决win11更新后闪屏,任务栏消失以及鼠标箭头不断刷新的bug
提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 前言 解决win11更新后电脑闪屏.鼠标不断刷新以及任务栏消失的问题 一.注意 二.解决办法 前言 解决win11更新后电脑 ...
- easyx图形库-----贴图技巧之双缓冲消除闪屏(BeginBatchDraw 与 EndBatchDraw)
目录 前言: 案例1:作图时闪屏 案例2:贴图时闪屏 双缓冲解决闪屏问题 原理: 前言: 学了easyx图形库的朋友们都知道,我们可以在easyx图形库上面画出连续运动的图片,或者通过贴图的方式把每一 ...
- c#双缓冲绘图(不闪烁的几种方法)
C#绘图双缓冲 C#双缓冲解释: 简单说就是当我们在进行画图操作时,系统并不是直接把内容呈现到屏幕上,而是先在内存中保存,然后一次性把结果输出来,如果没用双缓冲的话,你会发现在画图过程中屏幕会闪的很厉 ...
最新文章
- 干货 | 使用FFT变换自动去除图像中严重的网纹
- mybatisplus 结果_springboot整合mybatisPlus 乐观锁的实现
- Java EE + MongoDb与Apache TomEE和Jongo Starter项目
- js获取浏览器滚动条距离顶端的距离
- react学习(32)----onref
- 几十行python代码构建一个前后端分离的目标检测演示网站,代码开源
- Vim新手必看:Vim 命令图解
- JavaScript原生Ajax
- 卫星移动通信系统的分类
- python -- 判断给定的参数是否是地理位置的经度和纬度
- 继续学习-CSS3页面美化之静态美化
- echats统计图表的设计与实现
- 学习记录 Halcon 图片拼接
- MongoDB安装中使用配置文件的方式启动服务时出现错误:forked process: 2784 ERROR: child process failed, exited with 100
- 电容降压 20170619 周一
- WPF --三维空间(一)(简介)
- python在地图上画路线_使用Python和Perl绘制北京跑步地图
- linux怎么滑动命令行窗口_如何在Linux命令行界面愉快进行性能测试
- 浅谈应试教育与信息时代
- 第三届全国网络空间安全技术大赛 Web补题 By Assassin(持续更新)
热门文章
- stm32链接电脑提示无法识别的驱动设备
- yolov配置之:cuda、 cudnn安装
- 【CV夏季划】2021年冲刺CV秋招,100余课时从理论基础到进阶实践系统掌握
- 【知识星球】做作业还能赢奖金,传统图像/机器学习/深度学习尽在不言中
- 计算机生活工作原理,计算机基本工作原理是什么?
- java如何把文件中的内容存到一个动态数组arraylist中_如何动态地向Java中的数组添加项目?...
- JUnit 测试含有控制台输入的方法
- 2.Linux技能要求
- linux挂载和卸载
- ssh代理登录内网服务器