在窗體的構造函數中添加代碼:

SetStyle(ControlStyles.UserPaint, true); SetStyle(ControlStyles.AllPaintingInWmPaint, true); // 禁止擦除背景. SetStyle(ControlStyles.DoubleBuffer, true); // 雙緩沖
然后在構造函數下面,寫以下方法:

#region 解決閃爍問題 protected override void WndProc(ref Message m) { if (m.Msg == 0x0014) // 禁掉清除背景消息 return; base.WndProc(ref m); } protected override CreateParams CreateParams { get { CreateParams cp = base.CreateParams; cp.ExStyle |= 0x02000000; return cp; } } #endregion
這種方法的確可以解決閃屏問題,在電腦上切換沒有問題,但是在觸控機上切換會有黑色的一條條,暫時還未解決,就是第一次進來的時候有,希望看到此貼的人,如果有什么好的方法,歡迎下方評論,謝謝
最近對代碼作了一些優化,試驗后效果還可以,但是發現界面會閃爍,具體是TreeView控件會閃爍,語言為C#,IDE為VS2005。在查閱一些資料,使用了一些基本技術后(如開啟雙緩沖),發現沒什么效果。
於是使用Profiler工具,查找出瓶頸在於每次更新完界面的EndUpdate操作(使用這個是為了減少界面更新次數,但這里不理想是因為控件中中的元素很多),猜想大概每次更新,.Net底層都會更新重繪每個圖元,所以速度會慢,造成閃爍。但是如果這樣,使用雙緩沖應該會有較好效果。再看代碼,發現可能是更新動作太過頻繁,於是降低速度,有所好轉,但還是不行。
繼續在網上查閱,最終找到一個方案比較合適。原來底層重繪每次會清除畫布,然后再全部重新繪制,這才是導致閃爍最主要的原因。於是重載消息發送函數操作,禁掉這條消息。代碼如下:
protected override void WndProc(ref Message m)
{
if (m.Msg == 0x0014) // 禁掉清除背景消息
return;
base.WndProc(ref m);
}
成功!
注:雙緩沖還是有用的,在更新不是很頻繁且控件內含元素不是特別多的時候。一旦元素過多,每次更新時間都比較長,即便使用了雙緩沖,仍解決不了閃爍問題。個人認為最終比較理想的方法還是禁掉清除背景消息。
附:一些嘗試過但失敗的記錄
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)
{
// 此處書寫過濾消息代碼
}
但是實際實驗顯示無效果,不知是什么原因,沒有細究。
我的操作系統是Win7,使用的VS版本是VS2012,文中的代碼都是C#代碼。
這幾天遇到一個問題,即我用一個嵌入圖片的Panel作為Winform應用程序的背景,如下圖所示:
這是一個Winform窗體,里面放置了一個Panel,Dock屬性為Fill,BackgroundImage使用了《少年電世界》2003年第02期的封面圖片,BackgroundImageLayout使用了Stretch。
這個界面現在有兩個問題:
1、在窗體第一次被打開時,背景圖片會出現明顯的閃爍
2、在拉動窗體的邊界以調整窗體大小時,背景圖片非出現明顯的閃爍
為了處理這一問題,我查了一些資料,也都逐個試過了,下面先說下其中的兩個有代表性方法:
方法1:直接使用雙緩沖
-
SetStyle( ControlStyles.UserPaint, true);
-
SetStyle( ControlStyles.AllPaintingInWmPaint, true); // 禁止擦除背景.
-
SetStyle( ControlStyles.DoubleBuffer, true); // 雙緩沖
我嘗試着將這段代碼加到窗體的構造函數中,並不能解決問題,閃爍依然非常明顯
在MSDN上還有一篇文章《如何通過對窗體和控件使用雙緩沖來減少圖形閃爍》
地址:https://msdn.microsoft.com/zh-cn/library/3t7htc9c%28v=vs.80%29.aspx
這篇文章中也介紹了一個方法使用雙緩沖:
SetStyle(ControlStyles.OptimizedDoubleBuffer, true);
這個方法依然不能解決問題。
方法2:重寫CreateParams方法
方法2需要將以下這段代碼放在Form類的代碼內:
-
protected override CreateParams CreateParams
-
{
-
get
-
{
-
CreateParams paras = base.CreateParams;
-
paras.ExStyle |= 0x02000000;////用雙緩沖繪制窗口的所有子控件
-
return paras;
-
}
-
}
這個方法我一開始嘗試的時候一度認為是有效的,但使用了一段時間后還是發現了問題:
1、這個方法可以解決問題1,但不能解決問題2
2、這個方法會影響一些其他控件、組件的重繪(這點才是致命的)
因此,這個方法也不能解決問題。
上面兩個方法都不能解決問題,於是我繼續求助度娘,終於在下面這個頁面找到了解決方法:
方法3:封裝Panel類
http://blog.chinaunix.net/uid-14414741-id-2814313.html
這個方法,需要新建一個PanelEnhanced類繼承Panel類,代碼如下:
-
/// <summary>
-
/// 加強版 Panel
-
/// </summary>
-
class PanelEnhanced : Panel
-
{
-
/// <summary>
-
/// OnPaintBackground 事件
-
/// </summary>
-
/// <param name="e"></param>
-
protected override void OnPaintBackground(PaintEventArgs e)
-
{
-
// 重載基類的背景擦除函數,
-
// 解決窗口刷新,放大,圖像閃爍
-
return;
-
}
-
-
/// <summary>
-
/// OnPaint 事件
-
/// </summary>
-
/// <param name="e"></param>
-
protected override void OnPaint(PaintEventArgs e)
-
{
-
// 使用雙緩沖
-
this.DoubleBuffered = true;
-
// 背景重繪移動到此
-
if (this.BackgroundImage != null)
-
{
-
e.Graphics.SmoothingMode = System.Drawing.Drawing2D.SmoothingMode.HighQuality;
-
e.Graphics.DrawImage(
-
this.BackgroundImage,
-
new System.Drawing.Rectangle(0, 0, this.Width, this.Height),
-
0,
-
0,
-
this.BackgroundImage.Width,
-
this.BackgroundImage.Height,
-
System.Drawing.GraphicsUnit.Pixel);
-
}
-
base.OnPaint(e);
-
}
-
}
將之前我們建立窗體中的Panel容器換為我們新封裝的PanelEnhanced容器,將程序的背景圖片放到里面,再運行程序,程序背景閃爍的問題就完美解決了!