您现在的位置: 365建站网 > 365文章 > 解决C#/vb.net winform中的panel重绘导致界面闪烁问题

解决C#/vb.net winform中的panel重绘导致界面闪烁问题

文章来源:365jz.com     点击数:3635    更新时间:2018-06-22 21:07   参与评论

首先我们需要重新创建一个Panel类,其继承系统自带的Panel类,然后充新写一个构造函数,对其中的部分样式进行更改。

代码:


</>code

  1. public class NewPanel:Panel  
  2. {  
  3.     public NewPanel()  
  4.     {  
  5.         this.SetStyle(ControlStyles.AllPaintingInWmPaint, true);  
  6.         this.SetStyle(ControlStyles.OptimizedDoubleBuffer, true);  
  7.         this.SetStyle(ControlStyles.UserPaint, true);  
  8.     }  
  9. }

  


然后在窗口初始化的代码块中更改为我们当前新建的Panel类即可,其它代码都不必用~。

个人认为是非常棒的一种解决方案,可以完全解决闪烁的问题。


利用winform开发时,可能都会遇到一个问题,就是在panel中不停的重绘图形时,图形会不停的闪烁。要解决这个办法只需要开启双缓冲即可,由于初学c#,理解的不是很深,所以不多做解释。以下代码亲测可以解决这个问题:

首先创建一个自己的panel类:

</>code

  1. using System;  
  2. using System.Collections.Generic;  
  3. using System.Linq;  
  4. using System.Text;  
  5. using System.Windows.Forms;  
  6.    
  7. namespace Test  
  8. {  
  9.     //开启双缓冲  
  10.     class MyPanel:Panel  
  11.     {  
  12.         public MyPanel()  
  13.         {  
  14.             SetStyle(ControlStyles.UserPaint | ControlStyles.AllPaintingInWmPaint | ControlStyles.OptimizedDoubleBuffer | ControlStyles.ResizeRedraw | ControlStyles.SupportsTransparentBackColor, true);  
  15.         }  
  16.     }  
  17. }

  


然后重新生成解决方法,就会发现工具箱中多了一个自己写的panel控件,在画图形的时候,画在自己的panel上就可以了。由于自己的panel继承自Panel类,所以Panel中的方法都可以使用。


vb.net版本:

</>code

  1. Imports System
  2. Imports System.Collections.Generic
  3. Imports System.Linq
  4. Imports System.Text
  5. Imports System.Windows.Forms
  6. Namespace Test
  7.     
  8.     '开启双缓冲   
  9.     Class MyPanel
  10.         Inherits Panel
  11.         
  12.         Public Sub New()
  13.             MyBase.New
  14.             SetStyle((ControlStyles.UserPaint  _
  15.                             Or (ControlStyles.AllPaintingInWmPaint  _
  16.                             Or (ControlStyles.OptimizedDoubleBuffer  _
  17.                             Or (ControlStyles.ResizeRedraw Or ControlStyles.SupportsTransparentBackColor)))), true)
  18.         End Sub
  19.     End Class
  20. End Namespace


另外技术文章说明:


一、通过对窗体和控件使用双缓冲来减少图形闪烁(当绘制图片时出现闪烁时,使用双缓冲)

对于大多数应用程序,.NET Framework 提供的默认双缓冲将提供最佳效果。默认情况下,标准 Windows 窗体控件是双缓冲的。可以通过两种方法对窗体和所创作的控件启用默认双缓冲。一种方法是将 DoubleBuffered 属性设置为 true,另一种方法是通过调用 SetStyle 方法将 OptimizedDoubleBuffer 标志设置为 true。两种方法都将为窗体或控件启用默认双缓冲并提供无闪烁的图形呈现。建议仅对已为其编写所有呈现代码的自定义控件调用 SetStyle 方法。

在构造函数里加上以下代码:

</>code

  1. this.DoubleBuffered = true;//设置本窗体
  2. SetStyle(ControlStyles.UserPaint, true);
  3. SetStyle(ControlStyles.AllPaintingInWmPaint, true); // 禁止擦除背景.
  4. SetStyle(ControlStyles.DoubleBuffer, true); // 双缓冲
  5. //SetStyle(ControlStyles.DoubleBuffer | ControlStyles.OptimizedDoubleBuffer | ControlStyles.AllPaintingInWmPaint, true);
  6. //UpdateStyles();

 

二、C#控件的闪烁问题解决方法总结
最近对代码作了一些优化,试验后效果还可以,但是发现界面会闪烁,具体是TreeView控件会闪烁,语言为C#,IDE为VS2005。在查阅一些资料,使用了一些基本技术后(如开启双缓冲),发现没什么效果。
        于是使用Profiler工具,查找出瓶颈在于每次更新完界面的EndUpdate操作(使用这个是为了减少界面更新次数,但这里不理想是因为控件中中的元素很多),猜想大概每次更新,.Net底层都会更新重绘每个图元,所以速度会慢,造成闪烁。但是如果这样,使用双缓冲应该会有较好效果。再看代码,发现可能是更新动作太过频繁,于是降低速度,有所好转,但还是不行。
       继续在网上查阅,最终找到一个方案比较合适。原来底层重绘每次会清除画布,然后再全部重新绘制,这才是导致闪烁最主要的原因。于是重载消息发送函数操作,禁掉这条消息。代码如下:

</>code

  1. protected override void WndProc(ref Message m)
  2.  {
  3.      if (m.Msg == 0x0014) // 禁掉清除背景消息
  4.         return;
  5.      base.WndProc(ref m);
  6.  }

  

成功!

 

注:双缓冲还是有用的,在更新不是很频繁且控件内含元素不是特别多的时候。一旦元素过多,每次更新时间都比较长,即便使用了双缓冲,仍解决不了闪烁问题。个人认为最终比较理想的方法还是禁掉清除背景消息。


附:一些尝试过但失败的记录
1)使用setStyle
      网上有说使用setStyle函数去设置该控件的参数,具体为:
      SetStyle(ControlStyles.UserPaint | ControlStyles.AllPaintingInWmPaint | ControlStyles.OptimizedDoubleBuffer, true);
      这三个选项参数后者是依赖前者的,必须并存,否则无效。并且这个函数本身是protected的,所以首先需要继承某控件再使用。
      这个目标是跟前面正确解决方案一致,也是禁止清除背景并开启双缓冲,但需要使用用户绘制选项,而且是全部交由用户绘制。这需要自己实现控件的全部绘制,比较麻烦。所以这个方法不是完全不可行,但是需要额外工作量,不推荐。我也没有使用。


2)使用BeginUpdate和EndUpdate
      这一对操作对于需要批量操作更新控件的情景有比较好的效果,比如初始化时批量添加了大量节点。坏处就在于不能即时更新。所以,对于频繁的更新节点并希望立即反映到界面的情况不适用。如果使用并且没有禁掉清除界面消息的话,则控件看起来就会不停的闪烁,而且以白底为主,内容几乎不可见(这个视频繁程度而定)。因为界面更新都在EndUpdate处完成,操作太多导致EndUpdate阻塞时间过长,且清空在先,更新在后,导致界面看起来长时间处于空白状态。


3)使用ControlStyles.EnableNotifyMessage选项
      这个选项的作用和正确解决方案也是一致的。使用方法是:
      SetStyle(ControlStyles.EnableNotifyMessage, true);
      protected override void onNotifyMessage(Message m)
      {
               // 此处书写过滤消息代码
      }
      但是实际实验显示无效果,不知是什么原因,没有细究。

 

三、个人在一个winfrom中测试利用timer控件对要刷新的控件进行定时刷新,可能也能起到作用。

 

四、C# winform 局部刷新
做winform界面程序时,经常会遇到后台处理占用大量时间的情况,这就会造成界面假死状态。一般解决界面假死有两种方式:要么把占用大量时间的处理方式放入其他线程;要么把界面显示放入其他线程。第一种方式应该比较简单,开单独的线程,处理数据,将处理数据显示到界面就好。但是我们经常需要在主程序运算一些内容,否则可能会改动比较大。因此,这里讲讲第二种方式。
同样是使用多线程,但是c#在其他线程刷新有一点点问题,即不能跨线程操作界面。这可以使用控件的Invoke方法解决:

</>code

  1. private delegate void CrossThread();
  2.  Control control = ....;
  3.  CrossThread cross = delegate()
  4.  {
  5.       control.Refresh();
  6.  };
  7.  control.Invoke(cross);

这样可以让控件在其它线程刷新界面。


再加上开新线程后的通用方法:

</>code

  1.  private void InvaliateControl(Control control)
  2.   {
  3.       Thread t = new Thread(
  4.           new ThreadStart(delegate()
  5.           {
  6.               CrossThread cross = delegate()
  7.               {
  8.                   control.Refresh();
  9.               };
  10.              control.Invoke(cross);
  11.          }
  12.      ));
  13.  }

这样就可以在任何时候,调用此方法对控件进行刷新,而不将整个界面刷新。如果对于同一个控件,连续多次刷新,可以添加一个成员变量作为标记,以免同一控件连续多次刷新,提升部分性能。


补充:在主线程调用耗时操作用此方法可能会有问题,经过验证调用Invoke函数,其实是在主线程刷新界面。



如对本文有疑问,请提交到交流论坛,广大热心网友会为你解答!! 点击进入论坛

发表评论 (3635人查看0条评论)
请自觉遵守互联网相关的政策法规,严禁发布色情、暴力、反动的言论。
昵称:
最新评论
------分隔线----------------------------

快速入口

· 365软件
· 杰创官网
· 建站工具
· 网站大全

其它栏目

· 建站教程
· 365学习

业务咨询

· 技术支持
· 服务时间:9:00-18:00
365建站网二维码

Powered by 365建站网 RSS地图 HTML地图

copyright © 2013-2024 版权所有 鄂ICP备17013400号