C#Winform頻繁刷新致使界面閃爍解決方法
對於大多數應用程序,.NET Framework 提供的默認雙緩衝將提供最佳效果。默認狀況下,標準 Windows 窗體控件是雙緩衝的。能夠經過兩種方法對窗體和所創做的控件啓用默認雙緩衝。一種方法是將 DoubleBuffered 屬性設置爲 true,另外一種方法是經過調用 SetStyle 方法將 OptimizedDoubleBuffer 標誌設置爲 true。兩種方法都將爲窗體或控件啓用默認雙緩衝並提供無閃爍的圖形呈現。建議僅對已爲其編寫全部呈現代碼的自定義控件調用 SetStyle 方法。html
在構造函數里加上如下代碼:c#
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 7 //UpdateStyles();
2、C#控件的閃爍問題解決方法總結
最近對代碼做了一些優化,試驗後效果還能夠,可是發現界面會閃爍,具體是TreeView控件會閃爍,語言爲C#,IDE爲VS2005。在查閱一些資料,使用了一些基本技術後(如開啓雙緩衝),發現沒什麼效果。
因而使用Profiler工具,查找出瓶頸在於每次更新完界面的EndUpdate操做(使用這個是爲了減小界面更新次數,但這裏不理想是由於控件中中的元素不少),猜測大概每次更新,.Net底層都會更新重繪每一個圖元,因此速度會慢,形成閃爍。可是若是這樣,使用雙緩衝應該會有較好效果。再看代碼,發現多是更新動做太過頻繁,因而下降速度,有所好轉,但仍是不行。
繼續在網上查閱,最終找到一個方案比較合適。原來底層重繪每次會清除畫布,而後再所有從新繪製,這纔是致使閃爍最主要的緣由。因而重載消息發送函數操做,禁掉這條消息。代碼以下:多線程
1 protected override void WndProc(ref Message m) 2 { 3 if (m.Msg == 0x0014) // 禁掉清除背景消息 4 return; 5 base.WndProc(ref m); 6 }
成功!ide
注:雙緩衝仍是有用的,在更新不是很頻繁且控件內含元素不是特別多的時候。一旦元素過多,每次更新時間都比較長,即使使用了雙緩衝,仍解決不了閃爍問題。我的認爲最終比較理想的方法仍是禁掉清除背景消息。函數
附:一些嘗試過但失敗的記錄
1)使用setStyle
網上有說使用setStyle函數去設置該控件的參數,具體爲:
SetStyle(ControlStyles.UserPaint | ControlStyles.AllPaintingInWmPaint | ControlStyles.OptimizedDoubleBuffer, true);
這三個選項參數後者是依賴前者的,必須並存,不然無效。而且這個函數自己是protected的,因此首先須要繼承某控件再使用。
這個目標是跟前面正確解決方案一致,也是禁止清除背景並開啓雙緩衝,但須要使用用戶繪製選項,並且是所有交由用戶繪製。這須要本身實現控件的所有繪製,比較麻煩。因此這個方法不是徹底不可行,可是須要額外工做量,不推薦。我也沒有使用。工具
2)使用BeginUpdate和EndUpdate
這一對操做對於須要批量操做更新控件的情景有比較好的效果,好比初始化時批量添加了大量節點。壞處就在於不能即時更新。因此,對於頻繁的更新節點並但願當即反映到界面的狀況不適用。若是使用而且沒有禁掉清除界面消息的話,則控件看起來就會不停的閃爍,並且以白底爲主,內容幾乎不可見(這個視頻繁程度而定)。由於界面更新都在EndUpdate處完成,操做太多致使EndUpdate阻塞時間過長,且清空在先,更新在後,致使界面看起來長時間處於空白狀態。post
3)使用ControlStyles.EnableNotifyMessage選項
這個選項的做用和正確解決方案也是一致的。使用方法是:
SetStyle(ControlStyles.EnableNotifyMessage, true);
protected override void onNotifyMessage(Message m)
{
// 此處書寫過濾消息代碼
}
可是實際實驗顯示無效果,不知是什麼緣由,沒有細究。性能
3、我的在一個winfrom中測試利用timer控件對要刷新的控件進行定時刷新,可能也能起到做用。測試
4、C# winform 局部刷新
作winform界面程序時,常常會遇到後臺處理佔用大量時間的狀況,這就會形成界面假死狀態。通常解決界面假死有兩種方式:要麼把佔用大量時間的處理方式放入其餘線程;要麼把界面顯示放入其餘線程。第一種方式應該比較簡單,開單獨的線程,處理數據,將處理數據顯示到界面就好。可是咱們常常須要在主程序運算一些內容,不然可能會改動比較大。所以,這裏講講第二種方式。
一樣是使用多線程,可是c#在其餘線程刷新有一點點問題,即不能跨線程操做界面。這可使用控件的Invoke方法解決:優化
1 private delegate void CrossThread(); 2 Control control = ....; 3 CrossThread cross = delegate() 4 { 5 control.Refresh(); 6 }; 7 control.Invoke(cross);
這樣可讓控件在其它線程刷新界面。
再加上開新線程後的通用方法:
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函數,實際上是在主線程刷新界面。
原文連接:http://blog.csdn.net/weixin_40976261/article/details/78517409