winform的双缓冲

搜搜winform的双缓冲,就会发现网络上有很多文章,乱七八糟说的不明不白。第一种方案:

   

SetStyle(ControlStyles.UserPaint, true);
SetStyle(ControlStyles.AllPaintingInWmPaint, true); // 禁止擦除背景.
SetStyle(ControlStyles.DoubleBuffer, true); // 双缓冲

  第二种方案:

this.DouleBuffered=true

  第三种方案:

  Bitmap bimtBitmap = new Bitmap(width, height);
            Graphics g = Graphics.FromImage(bimtBitmap);
            //绘制区域
            g.Drawxxxx();
            g.DrawImage(bimtBitmap,this.ClientRectangle);

  当然还有其他,就是以上代码混杂着来,丝毫不清楚这些代码的具体含义,在此我解释一下,第一种方案=第二种方案=第三种方案+userPaint和AllPaintingInVmPaint设置为true。

首先说明一下为什么闪烁?比如说你要画两条线,但是写的绘制算法复杂度比较高,导致中间过程较慢,就会导致先出了一条,后出了一条,但是你的目标是一下子出现两条,所以就闪烁了,类似以下代码。

    protected override void OnPaint(PaintEventArgs e)
        {
            base.OnPaint(e);
            Pen pen = new Pen(Color.Red);
            e.Graphics.DrawLine(pen,0,0,this.Width,this.Height);
            Thread.Sleep(500);
            e.Graphics.DrawLine(pen,10,0,this.Width,this.Height);
           Thread.Sleep(500);

        }

  这种问题怎么解决呢? 1,2,3方案都可以解决。试试就会发现,不用上述一二三方案,线条一条一条出,用了之后肯定是两条一起出,无闪烁。但是闪烁并不是说唯一这种原因发生的,发生绘制之前还会绘制背景的,假如说OnPaintBackground和OnPaint之间时间过长呢?  背景把颜色擦成控件色,你再画,之间时间过长照样闪烁,此时能解决问题的是1,2。第三种方案解决不了问题,想让第三种方案解决问题得加上第一种方案前两行。

为什么会这样呢? 其实第三种方案是我们程序本身实现的双缓冲,但是假如OnPaintBackground和OnPaint之间时间过长,不论你在OnPaint里咋搞双缓冲,闪烁的是中间的间隔而已,解决不了问题。那1,2方案怎么就解决了?那我下边说明一下。看看第三种方案发生了什么。

首先当我们不开双缓冲的时候,在Control类的WndProc函数中,当消息号是20时,发生背景擦除,具体代码反编译即可。

这是一次完整的消息处理,之后会进行走OnPaint,消息号是15

那么这两次函数之间的间隔就会导致闪烁,不过一般程序也看不出来,假如是这种闪烁,人工模拟的双缓冲是解决不了问题的,那么为什么说1,2是等价方式呢?当我们开启双缓冲之后,看下Control类内代码:

protected virtual bool DoubleBuffered
    {
      get
      {
        return this.GetStyle(ControlStyles.OptimizedDoubleBuffer);
      }
      set
      {
        if (value == this.DoubleBuffered)
          return;
        if (value)
          this.SetStyle(ControlStyles.AllPaintingInWmPaint | ControlStyles.OptimizedDoubleBuffer, value);
        else
          this.SetStyle(ControlStyles.OptimizedDoubleBuffer, value);
      }
    }

  其实是开了AllPaintingInWmPaint ,这个是禁止背景擦除的,当我们设置这个为true之后,在上边的那个WmErasebkGnd函数中,是不会进行背景擦出的

那背景就不绘制了吗?当然不是,此时先收到15号消息:

进入这个函数之后:

首先看到一个flag1,这个其实是判断是否双缓冲的,可以看出来它是由DoubuleBuffered属性或者后边的this.GetStyle(ControlStyles.AllPaintingInWmPaint) && this.DoubleBufferingEnabled;而DoubleBufferingEnabled是什么呢?看源码:

所以flag1是DoubleBuffered属性|| UserPaint |DoubleBuffer|AllPaintingInWmPaint都为true的情况,所以我说1,2种两种方案其实是一致的,至于说1,2两种方案好于3的原因在于,1,2两种方案都禁止了擦除消息,然而在这个函数中是进行了背景绘制的,首先将Graphics保存起来,然后先画背景,再画Paint中代码,最后Render,是一次性绘制上去的,所以没有闪烁,而单纯用第三种方案,虽然自己实现了缓冲,然并卵,如果背景和前景时间过长照样闪烁,所以应该将ALLPaintInWmPaint设置了才能达到一样的效果。这个函数的具体相关代码如下:

至于UserPaint还有那些属性,文档上也含糊不清,其实多看源码就懂了,UserPaint几乎都是true,它不是true,OnPaint函数都不会走,Control类的构造里其实将UserPaint和AllPaintingInWmPaint都设置为true了,子类控件会有更改。AllPaintingInWmPaint可以让控件消息处理忽略擦除背景消息,控件的douleBuffered属性设置和你直接设置那三个枚举到达的是一个效果。

时间: 2024-11-08 19:51:32

winform的双缓冲的相关文章

为Winform配置双缓冲避免闪屏

最近新做了一个项目,项目的界面好多的控件,然后在进入界面的时候会刷新好多次,而且每次操作都有不必要的刷新,研究发现了需要配置双缓冲才能避免闪屏,代码如下. 1 //获取当前系统的环境版本,不同的环境对应不同的属性. 2 Version v = System.Environment.Version; 3 4 if (v.Major < 2) 5 { 6 this.SetStyle(ControlStyles.DoubleBuffer, true); 7 } 8 else 9 { 10 this.S

winform上控件太多,绘制时会逐个出现,通常说双缓冲能解决但实际不能解决的问题的解决方法。

protected override CreateParams CreateParams { get { CreateParams cp = base.CreateParams; cp.ExStyle |= 0x02000000; return cp; } } 接着找办法解决500多个控件PAINT缓慢的效能问题,心塞.分类都不想分了,就这样. winform上控件太多,绘制时会逐个出现,通常说双缓冲能解决但实际不能解决的问题的解决方法.,布布扣,bubuko.com

.net WINFORM的GDI双缓冲的实现

有时候在窗体中执行不断的GDI+操作的时候会出现闪速的状况,除了修改窗体的参数,更应该解决刷新本身的问题,双缓冲可能就是这样来的. 方法1: 用GDI绘制在位图上,然后再重新生成位图 Bitmap bt = new Bitmap(510, 510); Graphics g = Graphics.FromImage(bt); g.DrawEllipse(Pens.Red, 100, 100, new Random().Next(1, 60), 50); g.Dispose(); gb.DrawIm

一行代码开启 Winform 中的 ListView 和 DataGridView 双缓冲功能

ListView 中的 DoubleBuffered 属性是 protected 的,外部不能直接修改,但实际使用中,我们一般都需要开启双缓冲来避免控件闪烁. 网上很多人都是通过自定义 ListView 来实现的,虽然也不复杂,但其实用反射是更简单的办法: lv.GetType().GetProperty("DoubleBuffered", BindingFlags.Instance | BindingFlags.NonPublic).SetValue(dgv, true, null)

avalon与双缓冲技术

avalon与双缓冲技术 avalon1.5一个重要技术升级是引进异步渲染.异步渲染在游戏界有一个更专业的名字,叫双缓冲.游戏界要刷新界面与我们刷新浏览器视图,面临的问题是一致的.视图是由许多存在套嵌关系的方块组成,它们每一个的改动,都可能引起reflow(其父节点,其父父节点的大小重新计算),这是造成性能问题的关键. 双缓冲技术的主要原理是:当一个动画争先显示时,程序又在改变它,前面的画面还没显示完,程序又要求重新绘制,这样屏幕就会不停闪烁.为了避免闪烁,可以使用双缓冲技术,将要处理的图片都放

MFC利用双缓冲刷新绘图

在VC中进行绘图过程处理时,如果图形刷新很快, 经常出现图形闪烁的现象.利用先在内存绘制,然后 拷贝到屏幕的办法可以消除屏幕闪烁,具体的方法是先在内存 中创建一个与设备兼容的内存设备上下文,也就是开辟一快内 存区来作为显示区域,然后在这个内存区进行绘制图形.在绘制完成后利用 BitBlt函数把内存的图形直接拷贝到屏幕上即可. 具体想实现的是: 在Dialog客户区的一个图片控件(IDC_MAP)中绘制几个动态的点,如果不用双缓冲的技术,在屏幕刷新的时候会有闪烁的现象. CRect rect; C

OpenGL中实现双缓冲技术

在OpenGL中实现双缓冲技术的一种简单方法: 1.在调用glutInitDisplayMode函数时, 开启GLUT_DOUBLE,即glutInitDisplayMode(GLUT_RGB | GLUT_DOUBLE);.这里将我们惯用的GLUT_SINGLE替换为GLUT_DOUBLE,意为要使用双缓冲而非单缓冲. 2. 调用glutDisplayFunc(display)注册回调函数时, 在回调函数中所有绘制操作完成后调用glutSwapBuffers()交换两个缓冲区指针. 3. 调用

OpenGL的消隐与双缓冲

首先是大家可能已经发现,在我们之前提到的所有例子中,在图形的旋转过程中整个图形都有一定程度的闪烁现象,显得图形的过渡极不平滑,这当然不是我们所要的效果,幸好opengl 支 持一个称为双缓存的技术,可以有效的帮助我们解决这个问题.我们知道在我们电脑中,屏幕中显示的东西都会被放在一个称为显示缓存的地方,在通常情况下我们 只有一个这样的缓冲区,也就是单缓冲,在单缓冲中任何绘图的过程都会被显示在屏幕中,这也就是我们为什么会看到闪烁,而所谓双缓冲就是再这个显示的缓冲区 之外 再建立一个不显示的缓冲区,我

GDI双缓冲绘图

一.简介 在进行复杂图形绘制时,若直接在屏幕DC上进行绘制,则会出现明显的闪烁.闪烁产生的原因是当绘制的图形较为 复杂时,图形绘制过程中就被刷新到屏幕上,导致结果断断续续地显示出来.双缓冲绘图的原理是在另开辟一块内存用于绘制,当所有绘制工作完成后将内存数据一 次性拷贝到屏幕上. 双缓冲绘图步骤: 创建兼容DC(CreateCompatibleDC) 创建兼容位图(CreateCompatibleBitmap) 将兼容位图选入兼容DC(SelectObject) 在兼容DC中进行绘制工作 将兼容D