winform的双缓冲

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了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属性设置和你直接设置那三个枚举到达的是一个效果。

 

以上是关于winform的双缓冲的主要内容,如果未能解决你的问题,请参考以下文章

applet 中的双缓冲是如何工作的?

带有文本覆盖的双缓冲 AlphaBlend rect

Winforms 双缓冲

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

InnoDB的双写缓冲

基于c#的双缓冲技术画图