• 浅谈C#中的双缓冲


    在编程当中,或多或少会接触到图像编程,对于图像编程来说窗口闪烁是个常见的问题,当窗口有大量的复杂的图元数据需要重绘,或者拥有自定义控件中的窗口闪烁问题更是显而易见的。出现闪烁的原因有很多种,大部分原因主要是因为触发WM_PAINT消息时窗体进行了重绘操作,此过程先是用窗体的背景色擦除窗口表面,再把窗体的图像绘制上去,但是如果这2个操作不在同一时间段完成的话,就会先看到背景色(大部分为白色)接着才看到图像,这样就会出现我们所说的窗体闪烁现象。那么如何解决这个问题呢,解决方法有很多,其中有个比较好的方法(个人认为)就是采用双缓冲机制来绘图,基本上可以解决大部分的问题。

          双缓冲的原理:尽量快的输出图像,使输出在一个刷新周期内完成,如果输出内容很多比较慢,那么采用内存缓冲的方法,先把要输出的内容在内存准备好,然后一次性输出到窗体上,简单的说来就是在窗口刷新一次的过程中,让所有图元同时显示到窗口中。

         在C#中 .Net Framework为编程人员提供了很好的操作双缓冲的方法,为采用双缓冲机制绘制比较复杂的图像数据带来便捷。下面简单的介绍在C#中实现双缓冲的几种方法。

     一:利用默认的双缓冲

    (1)在应用程序中使用双缓冲的最简便的方法是使用 .NET Framework 为窗体和控件提供的默认双缓冲。通过将 DoubleBuffered 属性设置为 true。          

     
    1. this.DoubleBuffered=true;  

    (2)使用 SetStyle 方法可以为 Windows 窗体和所创作的 Windows 控件启用默认双缓冲,在窗体或者控件的构造函数中添加如下代码即可:

     
    1. SetStyle(ControlStyles.ResizeRedraw,true);  
    2. SetStyle(ControlStyles.OptimizedDoubleBuffer,true);  
    3. SetStyle(ControlStyles.AllPaintingInWmPaint,true);  

              或

     
    1. this.SetStyle(ControlStyles.ResizeRedraw |  
    2.               ControlStyles.OptimizedDoubleBuffer |  
    3.               ControlStyles.AllPaintingInWmPaint, true);  
    4. this.UpdateStyles();  

     注:

    .net1.1 和 .net 2.0 在处理控件双缓冲上是有区别的。
    .net 1.1 中,使用:this.SetStyle(ControlStyles.DoubleBuffer, true); 
    .net 2.0中,使用:this.SetStyle(ControlStyles.OptimizedDoubleBuffer, true);

     二:手动管理双缓冲

         在C# 中手动管理缓冲图像有2中方法,一种是利用单独开辟内存实现双缓冲这种传统的方法,还有一种是利用 .Net Framework 中独有的BufferedGraphicsContext类实现。

       方法一: 自己开辟一个缓冲区(如一个不显示的Bitmap对象),在其中绘制完成后,再一次性显示,代码如下:   

     
    1. //1、在内存中建立一块“虚拟画布”  
    2.   Bitmap bmp = new Bitmap(200,200);  
    3.   
    4. //2、获取这块内存画布的Graphics引用  
    5.   Graphics bufferGraphics = Graphics.FromImage(bmp);  
    6.   
    7. //3、在这块内存画布上绘图  
    8.   bufferGraphics.Clear(this.BackColor);  
    9. bufferGraphics.DrawRectangle(Pens.Black,0,0,bmp.Width -1,bmp.Height -1);  
    10. bufferGraphics.DrawEllipse(Pens.Red,10,10,100,50);  
    11. bufferGraphics.DrawLine(Pens.Green,10,100,100,200);  
    12.   
    13. //4、将内存画布画到窗口中  
    14.   using(Graphics g = e.Graphics)  
    15.   {  
    16.       g.DrawImage(bmp, 10, 10);  
    17.   }  
    18.     
    19. //5. 释放资源  
    20.   bmp.Dispose();  
    21. bufferGraphics.Dispose();     

      方法二:

          对于更高级的双缓存情形,可以使用 .NET Framework 类实现自己的双缓存逻辑。负责单独分配和管理图形缓冲区的类是BufferedGraphicsContext 类。每个应用程序都有自己的默认BufferedGraphicsContext 来管理此应用程序的所有默认双缓冲。提供调用Current 可以检索对此实例的引用。通过调用Allocate 方法可以创建与屏幕上的绘图图面关联的BufferedGraphics 类的实例。此方法创建一个与特定呈现图面(如窗体或控件)关联的BufferedGraphics 实例。创建 BufferedGraphics 实例后,可以将图形绘制到由该实例的Graphics 属性表示的缓冲区。 执行所有图形操作后,可通过调用Render 方法将缓冲区的内容复制到屏幕上。 以下代码把方法一实现的效果用此方法来实现:

    1. BufferedGraphicsContext currentContext = BufferedGraphicsManager.Current;  
    2.   
    3. BufferedGraphics myBuffer = currentContext.Allocate(e.Graphics,e.ClipRectangle);  
    4.   
    5. Graphics g = myBuffer.Graphics;  
    6.   
    7. g.Clear(this.BackColor);  
    8. g.DrawRectangle(Pens.Black, 10, 10, 200, 200);  
    9. g.DrawEllipse(Pens.Red, 10, 10, 100, 50);  
    10. g.DrawLine(Pens.Green, 10, 100, 100, 200);  
    11.   
    12. myBuffer.Render(e.Graphics);  //呈现图像至关联的Graphics  
    13.   
    14. myBuffer.Dispose();  
    15. g.Dispose();  
     

    至此,双缓冲问题解决,两种方式的实现效果都一样,笔者私以为第二种方法占有的内存很少,不会出现内存泄露!

  • 相关阅读:
    Python基础笔记(五)
    Python基础笔记(四)
    Python基础笔记(三)
    Python基础笔记(二)
    Python基础笔记(一)
    分页存储过程
    MD Test
    vue路由的配置技巧
    Echarts的使用与配置项
    js中call,apply,bind之间的区别
  • 原文地址:https://www.cnblogs.com/wanzhongjun/p/6262277.html
Copyright © 2020-2023  润新知