C#關於TreeView樹在節點數較多時總是會出現閃爍的問題方法記錄


         首先介紹下背景吧,問題如題,這個問題應該說困擾我大半年了(不是說我沒有請教大佬,不是說我沒有上網查過,之前在搜索時,總是沒有解決此問題~~),直到最近一次在在優化代碼時,再次上網查找,在發現搜索詞條”C#控件閃爍問題“,才有了下文。額額額,於是我記錄了下面得到片段。

         對於TreeView節點可以動態創建,而且數量還很大的時候,我們給其設置什么雙緩沖,使用EndUpdate操作啥的,都是沒有效果的。使用了下面的代碼即可解決問題;

        根據調式,我查找出瓶頸在於每次更新完界面的EndUpdate操作(使用這個是為了減少界面更新次數,但這里不理想是因為控件中中的元素很多),猜想大概每次更新,.Net底層都會更新重繪每個圖元,所以速度會慢,造成閃爍。但是如果這樣,使用雙緩沖應該會有較好效果。再看代碼,發現可能是更新動作太過頻繁,於是降低速度,有所好轉,但還是不行。

       繼續在網上查閱,最終找到一個方案比較合適。原來底層重繪每次會清除畫布,然后再全部重新繪制,這才是導致閃爍最主要的原因。於是重載消息發送函數操作,禁掉這條消息。代碼如下:

        protected override void WndProc(ref Message m)

        {

            if (m.Msg == 0x0014) // 禁掉清除背景消息WM_ERASEBKGND

                return;

            base.WndProc(ref m);

        }

        成功!

注意了,注意了,這里其實不只是TreeView控件有閃爍問題,在C#中只要控件的內容數量大,都會存在閃爍。產生閃爍原因如下:

  Windows在窗口的具體繪制之前,會發送WM_ERASEBKGND消息通知該窗口檫除背景。默認情況下,會以窗口的默認背景色清除窗口。

       WM_ERASEBKGND消息和WM_PAINT消息的另外一種含義:背景色與前景色

      可以這樣理解WM_ERASEBKGND消息和WM_PAINT消息:

          1.WM_ERASEBKGND消息用於通知系統或者程序員繪制背景色
          2.WM_PAINT消息用於通知程序員繪制前景色,比如在WM_PAINT中調用TextOut函數輸出文本

詳情可以看看這位網友的:https://blog.csdn.net/analogous_love/article/details/50039467

注:雙緩沖還是有用的,在更新不是很頻繁且控件內含元素不是特別多的時候。一旦元素過多,每次更新時間都比較長,即便使用了雙緩沖,仍解決不了閃爍問題。個人認為最終比較理想的方法還是禁掉清除背景消息。

附:一些嘗試過但失敗的記錄(下面的方法我沒有嘗試)

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)

      {

               // 此處書寫過濾消息代碼

      }

      但是實際實驗顯示無效果,不知是什么原因,沒有細究。

上面轉載網友的!僅為記錄下。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM