本文引自:VC窗口闪烁问题的解决

概述

一般的windows复杂的界面需要使用多层窗口而且要用贴图来美化,所以不可避免在窗口移动或者改变大小时候出现闪烁。

闪烁产生的原因

原因一:

如果熟悉显卡原理的话,调用GDI函数向屏幕输出的时候并不是立刻就显示在屏幕上,而是写到了显存里,显卡每隔一段时间把显存的内容输出到屏幕上,这就是刷新周期。一般显卡的刷新周期是 1/80秒左右,具体数字可以自己设置的。这样问题就来了,一般画图都是先画背景色,然后再把内容画上去,如果这两次操作不在同一个刷新周期内完成,那么给人的视觉感受就是,先看到只有背景色的图像,然后看到画上内容的图像,这样就会感觉闪烁了。

解决办法:

尽量快的输出图像,使输出在一个刷新周期内完成,如果输出内容很多比较慢,那么采用内存缓冲的方法,先把要输出的内容在内存准备好,然后一次输出到显存。要知道一次API调用一般可以在一个刷新周期内完成。

原因二:

复杂的界面有多层窗口组成,当windows在窗口改变大小的时候是先重画父窗口,然后重画子窗口,子父窗口重画的过程一般无法在一个刷新周期内完成,所以会呈现闪烁。我们知道父窗口上被子窗口挡住的部分其实没必要重画的。

解决办法:

给窗口加上风格WS_CLIPCHILDREN,这样父窗口上被子窗口挡住的部分就不会重画了。如果同级窗口之间有重叠,那么需要再加上WS_CLIPSIBLINGS风格。代码如下:

1 int CWndFlashDemo::OnCreate(LPCREATESTRUCT lpCreateStruct)
2 {
3 auto style = GetWindowLong(m_hWnd, GWL_STYLE);
4 style = style | WS_CLIPCHILDREN | WS_CLIPSIBLINGS;
5 SetWindowLong(m_hWnd, GWL_STYLE, style);
6 }

原因三:

有时候需要在窗口上使用一些控件,比如IE,当你的窗口改变大小时IE会闪烁,即使当前窗口已有WS_CLIPCHILDREN风格。原因在于该窗口的类风格有CS_HREDRAW 或者 CS_VREDRAW风格。这两个风格表示窗口在宽度或者高度发生变化的时候重绘窗口。但这可能影响IE闪烁。

解决办法:

注册窗口时不要使用这两个风格,如果窗口需要在改变大小的时候重绘,那么可以在响应WM_SIZE消息的时候调用RedrawWindow函数。

原因四:

界面上窗口很多,而且改变大小时很多窗口都要移动和改变大小,如果使用MoveWindow或者SetWindowPos两个API来改变窗口的大小和位置,由于他们是等待窗口重画完成后才返回,所以过程很慢,这样视觉效果就可能会闪烁。

解决办法:

使用以下API来处理窗口移动:

BeginDeferWindowPos、DeferWindowPos、EndDeferWindowPos函数。

先调用 BeginDeferWindowPos函数设定需要移动的窗口的个数。使用DeferWindowPos函数来移动窗口,这个API并不真的造成窗口移动,而是在调用EndDeferWindowPos函数一次性完成所有窗口的大小和位置的改变。

需要注意的是,BeginDeferWindowPos设定的个数一定要和实际的个数一致,否则可能造成崩溃。

其他原因:

1、将Invalidate函数替换为InvalidateRect函数

Invalidate()会导致整个窗口的图象重画,需要的时间比较长,而InvalidateRect()仅仅重画Rect区域内的内容,所以所需时间会少一些。

2、禁止系统搽除你的窗口

系统在需要重画窗口的时候会帮你用指定的背景色来搽除窗口。可是,也许需要重画的区域也许非常小。或者,在你重画这些东西之间还要经过大量的计算才能开始。这个时候你可以禁止系统搽掉原来的图象。直到你已经计算好了所有的数据,自己把那些需要搽掉的部分用背景色覆盖。

1 dc.FillRect(rect,&brush);

其中,rect是需要搽除的区域,brush是带背景色的刷子。

要禁止系统搽除你的窗口,可以重载OnEraseBkgnd()函数,让其直接返回TRUE就可以了。

1 BOOL CMyWin::OnEraseBkgnd(CDC* pDC)
2 {
3 return TRUE;
4 }

3、使用双缓冲

GDI+:

 1 RECT rc;
2 GetClientRect(g_hwnd,&rc);
3 Bitmap bmp(int(rc.right),int(rc.bottom));
4
5 Graphics bmpGraphics(&bmp);
6 bmpGraphics.SetSmoothingMode(SmoothingModeAntiAlias);
7
8 /*Drawing on bitmap*/
9 SolidBrush bkBrush(Color(0,0,0));
10 bmpGraphics.FillRectangle(&bkBrush,0,0,rc.right,rc.bottom);
11
12 /*Drawing on DC*/
13 Graphics graphics(hdc);
14 /*Important! Create a CacheBitmap object for quick drawing*/
15 CachedBitmap cachedBmp(&bmp,&graphics);
16 graphics.DrawCachedBitmap(&cachedBmp,0,0);

GDI:

 1 RECT rc;
2 GetClientRect(hwnd,&rc);
3 HDC hMemDc = CreateCompatibleDC(hdc);
4 HBITMAP hBmp = CreateCompatibleBitmap(hdc,rc.right,rc.bottom);
5 HBITMAP hOldBmp = (HBITMAP)SelectObject(hMemDc,hBmp);
6 //在此使用hMemDc进行 GDI 绘制
7 BitBlt(hdc,0,0,rc.right,rc.bottom,hMemDc,0,0,SRCCOPY);
8 SelectObject(hMemDc,hOldBmp);
9 DeleteObject(hBmp);
10 DeleteObject(hMemDc);

MFC窗口闪烁问题的更多相关文章

  1. MFC 窗口刷新防止闪烁方法

    防止窗口闪烁的方法 1.将Invalidate()替换为InvalidateRect(). Invalidate()会导致整个窗口的图象重画,需要的时间比较长,而InvalidateRect()仅仅重 ...

  2. MFC窗口重绘

    Invalidate()与 UpdateAllViews()有什么分别 Invalidate()是让程序重画窗口. UpdateAllViews()是在DOC/VIEW结构中, 当一个视图的数据改变后 ...

  3. 将Cocos2dX渲染到MFC窗口上

    引用:http://www.cnblogs.com/windeer/archive/2012/11/18/2767750.html 引言 现在智能手机已经慢慢进入大众化,移动类应用开始火爆起来,游戏类 ...

  4. 同时支持控制台和MFC窗口程序的APP

    BOOL CMyApp::InitInstance() { if ( m_bShowGui==FALSE ) { FILE *stream = NULL; AllocConsole(); // 开辟控 ...

  5. MFC窗口分割以及各窗口间的通讯

    一个偶然的机会又重新接触了MFC窗口的分割,自己结合资料重新写了一个窗口分割的程序,现将具体流程跟大家分享一下: 1.我们先创建一个MFC单文档类的程序,具体分割方式先将单文档整个客户区分成两行一列, ...

  6. 重写MFC窗口上的关闭按钮事件(SDI, MDI, Dialog)

    This piece of code demonstrate how to override WM_CLOSE event. 点击窗口关闭按钮,触发相关事件! 有时候,在MFC程序退出之前,我们通常会 ...

  7. 解决direct2d拖拽窗口闪烁

    响应WM_ERASEBKGND,在OnEraseBkgnd()处返回FALSE,阻止GDI重绘客户区背景色,设置背景色的工作交给Direct2D在Render时设置,否则在Resize时会出现窗口闪烁 ...

  8. 原来MFC窗口样式随字符集而改变

    以前好像发现,MFC窗口上按钮的自动样式有时是有亮色边框3D效果的,有时没有,不知道原因,也没有追究,今天正好有机会发现了原因,原来是随字符集而改变的. 1.Unicode版本下的窗口 2.未设置的窗 ...

  9. 利用CWinThread实现跨线程父子MFC窗口

    利用CWinThread实现跨线程父子MFC窗口 MFC对象只能由创建该对象的线程访问,而不能由其他线程访问. 不遵守该准则将导致断言(assertion)或者无法预知的程序行为等运行期错误. 在多线 ...

  10. 第十一篇 -- 如何实现MFC窗口的最大化以及控件随最大化

    这一篇介绍的是怎么实现MFC窗口的最大最小化,以及里面控件大小也随之改变 第一步:实现窗口最大最小化 首先右击窗口空白处,打开properties,将里面的MaximizeBox和MinimizeBo ...

随机推荐

  1. 安川YASKAWA机器人主板维修方法合集

    安川机械手板卡故障分析与YASKAWA机械臂主板维修步骤 1. 确认故障现象:首先,我们需要详细了解安川机器人主板故障现象,包括但不限于工作异常.运行错误.速度变慢等. 2. 拆卸主板:根据故障现象, ...

  2. 最优化算法动量法Momentum

    动量法的结论: 1.动量方法主要是为了解决Hessian矩阵病态条件问题(直观上讲就是梯度高度敏感于参数空间的某些方向)的. 2.加速学习 3.一般将参数设为0.5,0.9,或者0.99,分别表示最大 ...

  3. Blazor Hybrid适配到HarmonyOS系统

    1. 前言 Blazor Hybrid是一个基于Web技术的MVVM开发模式的客户端框架,它只有UI是由Webview渲染,而逻辑均由.NET开发,可以突破浏览器限制访问本地文件,或者发起TCP或者U ...

  4. manim边学边做--局部缩放的场景类

    在动画制作中,尤其是数学和科学可视化领域,有时我们需要将观众的注意力集中在场景的某个特定部分. Manim提供了一个强大的工具 ZoomedScene,它允许我们在场景中创建一个独立的缩放视图,从而实 ...

  5. 探秘Transformer系列之(13)--- FFN

    探秘Transformer系列之(13)--- FFN 目录 探秘Transformer系列之(13)--- FFN 0x00 概述 0x01 网络结构 1.1 数学表示 1.2 中间层比率 1.3 ...

  6. mysql 连接失败:message from server: "Host '192.168.xx.xxx' is not allowed to connect to this MySQL server"

    前言 mysql 连接失败:message from server: "Host '192.168.xx.xxx' is not allowed to connect to this MyS ...

  7. $GOPATH/go.mod exists but should not

    开启模块支持后,并不能与GOPATH共存,所以把项目从GOPATH中移出即可

  8. HTTP/1.1 优化

    避免发送 HTTP 请求 对于一些具有重复性的 HTTP 请求,比如每次请求得到的数据都一样的,我们可以把这对「请求-响应」的数据都缓存在本地,通过缓存技术减少请求次数. 客户端会把第一次请求以及响应 ...

  9. 使用Win32控制台实现libevent通信

    libevent版本:libevent-2.0.22-stable 服务端: #include <string.h> #include <errno.h> #include & ...

  10. datasnap的监督功能【2】-管理Session

    1.服务端的Session是有TDSSession定义的.TDSSession提供了许多有用的方法和特性,再开发室取得服务or重要信息. 如Session状态.安排Session独享定时or自动执行工 ...