最近對(duì)代碼作了一些優(yōu)化,試驗(yàn)后效果還可以,但是發(fā)現(xiàn)界面會(huì)閃爍,具體是TreeView控件會(huì)閃爍,語(yǔ)言為C#,IDE為VS2005。在查閱一些資料,使用了一些基本技術(shù)后(如開啟雙緩沖),發(fā)現(xiàn)沒什么效果。
于是使用Profiler工具,查找出瓶頸在于每次更新完界面的EndUpdate操作(使用這個(gè)是為了減少界面更新次數(shù),但這里不理想是因?yàn)榭丶兄械脑睾芏啵,猜想大概每次更新?Net底層都會(huì)更新重繪每個(gè)圖元,所以速度會(huì)慢,造成閃爍。但是如果這樣,使用雙緩沖應(yīng)該會(huì)有較好效果。再看代碼,發(fā)現(xiàn)可能是更新動(dòng)作太過(guò)頻繁,于是降低速度,有所好轉(zhuǎn),但還是不行。
繼續(xù)在網(wǎng)上查閱,最終找到一個(gè)方案比較合適。原來(lái)底層重繪每次會(huì)清除畫布,然后再全部重新繪制,這才是導(dǎo)致閃爍最主要的原因。于是重載消息發(fā)送函數(shù)操作,禁掉這條消息。代碼如下:
protected override void WndProc(ref Message m)
{
if (m.Msg == 0x0014) // 禁掉清除背景消息
return;
base.WndProc(ref m);
}
成功!
注:雙緩沖還是有用的,在更新不是很頻繁且控件內(nèi)含元素不是特別多的時(shí)候。一旦元素過(guò)多,每次更新時(shí)間都比較長(zhǎng),即便使用了雙緩沖,仍解決不了閃爍問題。個(gè)人認(rèn)為最終比較理想的方法還是禁掉清除背景消息。
附:一些嘗試過(guò)但失敗的記錄
1)使用setStyle
網(wǎng)上有說(shuō)使用setStyle函數(shù)去設(shè)置該控件的參數(shù),具體為:
SetStyle(ControlStyles.UserPaint | ControlStyles.AllPaintingInWmPaint | ControlStyles.OptimizedDoubleBuffer, true);
這三個(gè)選項(xiàng)參數(shù)后者是依賴前者的,必須并存,否則無(wú)效。并且這個(gè)函數(shù)本身是protected的,所以首先需要繼承某控件再使用。
這個(gè)目標(biāo)是跟前面正確解決方案一致,也是禁止清除背景并開啟雙緩沖,但需要使用用戶繪制選項(xiàng),而且是全部交由用戶繪制。這需要自己實(shí)現(xiàn)控件的全部繪制,比較麻煩。所以這個(gè)方法不是完全不可行,但是需要額外工作量,不推薦。我也沒有使用。
2)使用BeginUpdate和EndUpdate
這一對(duì)操作對(duì)于需要批量操作更新控件的情景有比較好的效果,比如初始化時(shí)批量添加了大量節(jié)點(diǎn)。壞處就在于不能即時(shí)更新。所以,對(duì)于頻繁的更新節(jié)點(diǎn)并希望立即反映到界面的情況不適用。如果使用并且沒有禁掉清除界面消息的話,則控件看起來(lái)就會(huì)不停的閃爍,而且以白底為主,內(nèi)容幾乎不可見(這個(gè)視頻繁程度而定)。因?yàn)榻缑娓露荚贓ndUpdate處完成,操作太多導(dǎo)致EndUpdate阻塞時(shí)間過(guò)長(zhǎng),且清空在先,更新在后,導(dǎo)致界面看起來(lái)長(zhǎng)時(shí)間處于空白狀態(tài)。
3)使用ControlStyles.EnableNotifyMessage選項(xiàng)
這個(gè)選項(xiàng)的作用和正確解決方案也是一致的。使用方法是:
SetStyle(ControlStyles.EnableNotifyMessage, true);
protected override void onNotifyMessage(Message m)
{
// 此處書寫過(guò)濾消息代碼
}
但是實(shí)際實(shí)驗(yàn)顯示無(wú)效果,不知是什么原因,沒有細(xì)究。