我是搞控制項開發的,經常被人問,所以把一些問題記錄了下來!如果有人再問,直接把地址丟給他看。 一、 經常會有人抱怨Winform界面閃爍,下麵有幾個方法可以儘可能的避免出現閃爍 1.控制項的使用儘量以純色為主,儘量不使用背景圖,或者把大圖改成小圖,或者圖片不縮放繪製,或者直接用不透明的純色背景色(Win ...
我是搞控制項開發的,經常被人問,所以把一些問題記錄了下來!如果有人再問,直接把地址丟給他看。
一、 經常會有人抱怨Winform界面閃爍,下麵有幾個方法可以儘可能的避免出現閃爍
1.控制項的使用儘量以純色為主,儘量不使用背景圖,或者把大圖改成小圖,或者圖片不縮放繪製,或者直接用不透明的純色背景色(Winform控制項透明是偽透明,是通過截取父容器背景來實現的,所以效率比較低,而且有時候會有延遲。而且多個透明控制項疊加的時候不能分層顯示出各個控制項的透明效果)
2.自定義控制項中使用雙緩衝。
3.窗體邊框最好不要設置為None,保留窗體最大化最小化動畫,可以避免最小化恢復時候的界面閃爍。可能會有人說我的窗體是自定義美化的,邊框為None了。其實可以通過攔截消息重繪非客戶區,實現邊框不為None的時候重繪整個窗體的。 網上也有很多教程
二、自定義控制項時候需要註意的
1.控制項重繪的時候儘可能的處理e.ClipRectangle,減少不必要的繪圖操作,標準重繪都應該在Paint事件里處理,用事件里的e.Graphics。
而不是在隨便個地方就用CreateGraphics()來繪製,需要刷新的時候調用 Invalidate() 如果能計算重繪區域最好
2.使用雙緩衝,在構造函數裡加入這兩行代碼 減少閃爍
SetStyle(ControlStyles.OptimizedDoubleBuffer, true);
SetStyle(ControlStyles.AllPaintingInWmPaint, true);
或者
DoubleBuffered = true;
3.如果你開發dll組件給別人用,記得項目屬性里,生成,選擇 XML文檔文件,生成///<summary> 代碼註釋。到時候把dll給別人用的時候也把Xml帶上,放到和dll同一個目錄,這樣別人開發的時候就有註釋提示了
三、其他開發問題
1.多線程操作控制項請使用控制項的Invoke或者BeginInvoke調用
不要總想著其他線程中直接操作控制項,會出現很多奇葩問題的。
Control.CheckForIllegalCrossThreadCalls 這個屬性就是掩耳盜鈴,就和 SetProcessWorkingSetSize(將記憶體強制壓到虛擬記憶體里,給硬碟增加壓力) 用來釋放記憶體一樣。當初不知道誰傳出來的,誤導一群人!
正常的多線程操作控制項是
this.Invoke(new MethodInvoker(() => {
//UI操作代碼
}));
或者用 async 語法 非同步編程
不過還有人用很古老的寫法,定義委托,判斷 InvokeRequired在調用Invoke,雖然也可以,但是浪費時間。
還有 Application.DoEvents(); 也是個遠古代碼,能不用就不要用
使用線程的時候記得設置線程的 IsBackground=true; 程式關閉之後,線程能自動退出
2. 顏色System.Drawing.Color 這個結構體 是可以支持設置透明度的。可以3個或者4個值,Color.FromArgb(),4個值的時候是 Alpha,Red,Green,Blue. 第一個就是透明度的參數 0-255
3.如果你想把窗體最大化的時候全屏,遮住任務欄,請把窗體邊框設置為None
4.控制項佈局的時候活用Dock和Anchor,這兩個屬性是互斥的
5.單行的TextBox的高度受字體影響,不能單獨改。如果想美化邊框效果,請把TextBox的邊框隱藏了,放到Panel里,Panel在加個邊框背景美化。
6.Label換行只要有\r\n就會換行的,如果要自動換行,先把AutoSize改成false,設置好寬度。字元超過寬度就會換行的。
7.System.Windows.Forms.Timer 這個是執行在主線程的,本質是控制項消息迴圈里的一個定時消息,不要用來做耗時的操作。耗時操作請用 System.Threading.Timer和 System.Timers.Timer
8.Winform 對DPI支持的不完善,雖然.NET Framework 4.6 中有EnableWindowsFormsHighDpiAutoResizing。但是你必須所有控制項用原生控制項。
如果你用的不是.Net.4.6 而且使用了第三方控制項,為了不影響佈局,建議把裡面的字體單位都改成像素,容器控制項包括窗體的AutoScaleMode屬性改成None。這樣至少界面佈局基本正常,高DPI里只是模糊了些
9.使用非托管資源最好顯式調用Dispose釋放資源,比如圖片,文件讀取等等。
this.BackgroundImage = global::WinFormsTest.Properties.Resources.bt;
Properties.Resources這個對象的圖片屬性,每次訪問會創建一個圖片副本,如果多個控制項使用同一張圖片,建議手動寫代碼設置,先設置到一個靜態欄位里,再引用這個圖片欄位。
10. try 不要亂用,能解決問題的能不加try的就不要加try,太多try容易導致異常不好定位。一般對於不確定是否異常的才需要加try,比如文件讀取,網路連接等等,另外一種是一般開發底層框架的可以確定異常類型,可以配合throw new Exception,用來提示開發者調用出錯的信息。另外異常可以全局捕獲在 Application.Run加try和ThreadException事件記錄日誌。
之前看到有些人寫一大堆try,程式運行靠try維持,簡單的錯誤沒有判斷過濾掉,靠try來忽略。導致整個程式卡的一比!
四、一些VS快捷鍵
1.代碼+Tab+tab VS代碼快速生成補全,比如 if 再按兩下Tab就可以生成if(){}。 mb + Tab+tab= MessageBox.Show("Test"); try、switch、事件綁定+=等等。很多代碼都可以用這種方式快速生成。
2. F12 轉定義,Ctrl+R+R 重命名, Ctrl + F 除了查找之外,還可以做正則表達式的簡單驗證,因為匹配結果可以染色。
其他問題自己先去百度吧!
最後問一下:Winform還有多少人用,很多人都轉WPF了吧?另外.net的跨平臺UI有沒有意義?