XAML: 自定義控制項中事件處理的最佳實踐

来源:https://www.cnblogs.com/wpinfo/archive/2018/02/06/xaml_control_event_bp.html
-Advertisement-
Play Games

在開發 XAML(WPF/UWP) 應用程式中,有時候,我們需要創建自定義控制項 (Custom Control) 來滿足實際需求。而在自定義控制項中,我們一般會用到一些原生的控制項(如 Button、TextBox 等)來輔助以完成自定義控制項的功能。 自定義控制項並不像用戶控制項 (User Control ...


在開發 XAML(WPF/UWP) 應用程式中,有時候,我們需要創建自定義控制項 (Custom Control) 來滿足實際需求。而在自定義控制項中,我們一般會用到一些原生的控制項(如 Button、TextBox 等)來輔助以完成自定義控制項的功能。

自定義控制項並不像用戶控制項 (User Control) 一樣,使用 Code-Behind(UI 與邏輯在一起)技術。相反,它通過把 UI 與邏輯分離而將兩者解耦。因此,創建一個自定義控制項會產生兩個文件,一個是 Generic.xaml,在它裡面定義其模板與樣式;另一個是 <ControlName>.cs,這裡面存放其邏輯,如下圖:

在這種情況下,要想在代碼中獲取到模板里定義的控制項,就不像 Code-Behind 中那麼容易,而要藉助於 OnApplyTemplate 和 GetTemplateChild 這兩個方法。它們的意義分別如下:

  • OnApplyTemplate: 在自定義控制項中,通常要重寫這個方法,當基類調用 ApplyTemplate() 方法以構造可視化樹時,會調用它;
  • GetTemplateChild: 獲取 ControlTemplate 中所定義的可視化樹上指定名稱的元素;

所以,如果我們在模板中定義了一個名為 PART_ViewButton 的按鈕,那麼,我們可以這樣獲取它,併為它註冊響應事件:

        public override void OnApplyTemplate()
        {
            base.OnApplyTemplate();

            Button btnView = GetTemplateChild("PART_ViewButton") as Button;
            if (btnView != null)
            {
                btnView.Click += BtnView_Click;
            }
        }

        private void BtnView_Click(object sender, RoutedEventArgs e)
        {
            // 這裡寫響應邏輯
        }

當我們(或者其他人)要用這個控制項時,通過給它設置了模板(一般都是預設模板)後, OnApplyTemplate 方法就會被執行。這樣做看起來沒什麼問題。不過,其實這裡有可能會引起一個聽起來很嚴重的問題:記憶體泄露 (Memory Leak)

何為記憶體泄露

記憶體泄露有多種類型,一般來說,它是指某種類型的資源不再使用,但卻仍然占用記憶體。換句話說,它從受管理的記憶體區域中“泄漏”出去了,無法被 GC 回收。如果在程式中有多處記憶體泄露,將會占有很多記憶體,並最終導到記憶體被耗盡。

在 C# 中,常見的記憶體泄露有:

• 沒有移除事件監聽;
• 沒有銷毀非托管資源(如資料庫、文件流等);

對於上面兩種情況,它們的解決辦法也非常簡單,分別是:要反註冊事件(即移除事件監聽)與調用 Dispose 方法(如果沒有,則要實現 IDisposable 介面,併在其中銷毀非托管資源)。

對於第二種情況,比較好理解;而對於第一種情況,問題是,為什麼沒有移除事件監聽,會導致記憶體泄露呢?這是因為事件源比事件監聽者的生命周期更長。來看代碼:

            ObjectA objA = new ObjectA();
            ObjectB objB = new ObjectB();
            objA.Event += objB.EventHanlder;

ObjectA 中定義了 Event 事件,我們為它註冊了一個事件處理器(對象 objB 中的 EventHanlder 方法);因此,事件源 objA 對事件監聽對象 objB 存在一個引用。

如果 objB 不再使用,我們要銷毀它,但由於 objA 引用了它,所以它不會被銷毀、回收;它要等到 objA 銷毀時,才能被銷毀。所以本來需要被銷毀的對象,卻因有其它對象對它的引用,結果造成了記憶體泄露。

如何解決

再回到自定義控制項的問題上,因為我們的自定義控制項,可能會被重寫樣式或者重寫模板,這會使 OnApplyTemplate 方法在這個自定義控制項的生命周期內被執行多次。所以,我們需要為那些通過 GetTemplateChild 方法得到並且又添加了事件處理的控制項(如上述代碼中的 btnView 控制項)進行事件反註冊。因為這些都是前一個模板中的控制項(元素),當反註冊後,原來的控制項與事件監聽者(自定義控制項本身)就不存在引用關係,從而避免了記憶體泄露的問題。

根據我們的解決思路,對之前的代碼重構如下:

        private Button btnView = null;
        public override void OnApplyTemplate()
        {
            base.OnApplyTemplate();

            // 先反註冊事件
            if (btnView != null)
            {
                btnView.Click -= BtnView_Click;
            }

            btnView = GetTemplateChild("PART_ViewButton") as Button;

            if (btnView != null)
            {
                btnView.Click += BtnView_Click;
            }
        }

        private void BtnView_Click(object sender, RoutedEventArgs e)
        {
            // 這裡寫響應邏輯
        }

這樣,就解決了本文開頭所說的問題。不過,接下來,我們還需要做一點調整。

進一步重構

試想,如果我們的自定義控制項中,有多個類似像前述 btnView 這樣的控制項,我們就要將上面的代碼在 OnApplyTemplate 方法中複製若幹次,從而導致 OnApplyTemplate 方法的複雜度增加,以及代碼的可讀性變差 。

為了改善這一點,我們將每個控制項以及它的事件註冊與反註冊封裝一下。重構後,代碼如下:

        protected const string PART_ViewButton = nameof(PART_ViewButton);

        private Button btnView = null;

        public Button ViewButton
        {
            get
            {
                return btnView;
            }
            set
            {
                // 先反註冊事件
                if (btnView != null)
                {
                    btnView.Click -= BtnView_Click;
                }

                btnView = value;

                if (btnView != null)
                {
                    btnView.Click += BtnView_Click;
                }
            }
        }

        public override void OnApplyTemplate()
        {
            base.OnApplyTemplate();

            ViewButton = GetTemplateChild(PART_ViewButton) as Button;
        }

        private void BtnView_Click(object sender, RoutedEventArgs e)
        {
            // 這裡寫響應邏輯
        }

 針對最終的代碼,這裡再提幾點:

1. 在 OnApplyTemplate 方法中,建議一開始要先調用 base.OnApplyTemplate();
2. 無論在為控制項反註冊事件,還是註冊事件時,都要對控制項是否為空進行判斷,這是因為有可能用戶重寫模板時沒有遵循 TemplatePart 屬性中所指定的控制項名稱;
3. 將控制項的名稱聲明為常量,可以避免字元串拼寫錯誤; 

總結

本文討論了在 WPF 或 UWP 中創建自定義控制項時,可能會遇到記憶體泄露的問題;這主要是由於模板中的控制項事件沒有反註冊導致的。我們不僅分析了其中的原因,也給出了針對這種情況的最佳實踐。

雖然在一般情況下,這一問題並不會造成較大的影響,但是,如果我們能夠在這些細節上註意,這樣不僅能夠提高我們的代碼質量與程式的性能,也能夠給我們在設計或處理類似的問題時,提供必要的思路與經驗。

作者:WPInfo

本文系作者原創,歡迎轉載;如需轉載,請註明出處。

公眾號:.NET之窗 (WinDotNET),更多原創、優質技術文章,歡迎掃碼關註。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1.反射通過字元串映射或修改程式運行時的狀態、屬性、方法 getattr(obj,name_str): 根據字元串name_str去獲取obj對象里的對應的方法的記憶體地址 hasttr(obj,name_str): 判斷一個對象obj里是否有對應的字元串的方法 setattr(obj,'y',z): ...
  • 一、列表 支持的基本操作: 索引 切片 修改 刪除 迴圈 包含 1、append 2、clear 3、copy 4、count 5、extend 6、index 7、insert 8、pop 9、remove 10、reverse 11、sort 二、元祖 基本操作: 索引 切片 迴圈,可迭代對象 ...
  • #本文是在Windows環境下,Unix系統應該還要設置2個東西 (一) 採用MVC設計web應用 遵循 模型-視圖-控制器(model-view-controlle) 模型:存儲web應用數據的代碼 視圖:格式化和顯示web應用用戶界面的代碼 控制器:將web應用粘合在一起並提供業務邏輯的代碼 ( ...
  • 什麼是異常: 當程式遭遇某些非正常問題的時候就會拋出異常:比如int()只能處理能轉化成int的對象,如果傳入一個不能轉化的對象就會報錯並拋出異常 常用的異常有: ValueError :傳入無效的錯誤的參數 TypeError:進行了對類型無效的操作 IndexError:序列中沒有此索引 Nam... ...
  • 當類中的方法都是抽象方法, 介面格式特點: 1、介面中可以定義常量和抽象方法。 2、介面中成員有固定修飾符: 常量:public static final 可省略 方法:public abstract 可省略 3、介面中的成員都是public的。 4、子類實現介面需要使用 implements 關鍵 ...
  • 為什麼要使用 ASP.NET Core? .NET Core 剛發佈的時候根據介紹就有點心裡癢癢, 大概看了一下沒敢付諸於行動, 現在2.0發佈了一段時間了, 之前對其"不穩定"的顧慮也打消的差不多了, 決定踏實的研究一下. 至於為什麼要使用core, 官方是這樣說的: ASP.NET Core 是 ...
  • 在前面的文章中介紹了用戶的註冊及登錄功能,在註冊用戶時可以通過代碼的形式限制用戶名及密碼的格式,如果不符合要求那麼就無法完成操作,如下圖: 該功能的原理是Identity基於的Entity Framework組件在添加用戶之前對用戶提交數據進行校驗後給出的錯誤信息。 數據校驗功能在每一個軟體系統中都 ...
  • 最近閑來沒事研究了下12306網站的登錄,發現驗證碼其實不難破解,只要記錄正確圖片的具體坐標就好了。 具體登錄的實現只需要三步,而且全部是通過瀏覽器地址欄完成的噢!廢話不多說,現在開始三步走! 為使得更好操作,建議每一步打開一個新的標簽頁! 第一步:獲取圖片驗證碼 url:https://kyfw. ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...