C#之委托如此簡單

来源:https://www.cnblogs.com/cqhaibin/archive/2019/11/10/11831651.html

近期和幾位做嵌入式開發的朋友閑聊過程中,一位朋友抱怨到:這C#太難用了,我想在N個窗體(或者是N個用戶組件之間)傳遞值都搞不定,非得要定義一個全局變數來存儲,然後用定時器來刷新值,太Low了。我急切的回答道:這很簡單,不就是委托的事嘛。那你來一個示例啊:朋友道。此為這篇博客的起因,所以此篇博客對於有 ...


     近期和幾位做嵌入式開發的朋友閑聊過程中,一位朋友抱怨到:這C#太難用了,我想在N個窗體(或者是N個用戶組件之間)傳遞值都搞不定,非得要定義一個全局變數來存儲,然後用定時器來刷新值,太Low了。我急切的回答道:這很簡單,不就是委托的事嘛。那你來一個示例啊:朋友道。此為這篇博客的起因,所以此篇博客對於有c#開發經驗的伙伴們那是小菜一喋。

一、對委托的理解

     委托:同一個功能,可以根據不同的場景委托給不同的方法具體執行; 舉個慄子:某位美食愛好妹子,通常自己做美食;找到男票後,就男票做美食;換男票後,就第二任男票做美食。我們把這例子委托抽象化:

    定義一個委托功能:做美食;規範及流程:輸入”食材“,通過”做美食“的委托,輸出”美食“。

    委托實現之自己做:妹子自己做美食

    委托實現之一號男票做:一號男票做美食

    委托實現之二號男票做:二號男票做美食

    做美食這項功能,被妹子在不同的時間段分配給了不同的對象,雖然妹子,男一,男二做的美食都很好吃,但味道肯定有區別。這就是委托生活化的示例,各位看觀瞭解否(偷笑)。

二、代碼實現

    上面的示例如何用代碼實現,這裡就不展示了(真的很簡單)。下麵我們換一個稍有難度和實際應用的示例,需求說明:主窗體顯示一個列表,子窗體增加數據(不關閉子窗體的情況下),主窗體列表自動更新,且第二個子窗體打開後,窗體內的列表也要同時更新。

    UI設計:一個主窗體,兩個子窗體(A窗體:增加數據,B窗體:顯示數據),一個用戶組件(列表顯示內容)

2.1 EventBus實現

代碼如下:

public class EventBus<T>
    {
        private List<T> list = new List<T>();

        public event EventHandler<EventBusArg<List<T>>> EventNotice;
        public delegate void DelegateItemInfoEvent(List<T> items);

        public void Add(T item)
        {
            this.list.Add(item);
            this.TriggerEventNotice();
        }

        public void Remove(T item)
        {
            this.list.Remove(item);
            this.TriggerEventNotice();
        }

        public List<T> GetAll()
        {
            return this.list;
        }

        private void TriggerEventNotice()
        {
            if (this.EventNotice != null)
            {
                this.EventNotice.Invoke(this, new EventBusArg<List<T>>()
                {
                    Data = this.GetAll()
                });
            }
        }
    }

    public class EventBusArg<T> : EventArgs
    {
        public T Data { get; set; }
    }

重點:

1. 定義了一個委托類型:DelegateItemInfoEvent(List<T> items)

2. 定義了一個事件對象:EventHandler<EventBusArg<List<T>>>

3. 事件對象的參數必須繼承EventArgs對象

4. 事件依賴委托

2.2 主窗體

代碼如下:

 

        private EventBus<ItemInfo> eventBus = new EventBus<ItemInfo>();
        private EventBus<ItemInfo>.DelegateItemInfoEvent FunItem;
        public Form1()
        {
            InitializeComponent();
            this.eventBus.EventNotice += EventBus_EventNotice;
        }

        private void EventBus_EventNotice(object sender, EventBusArg<List<ItemInfo>> e)
        {
            if (this.ucList1.InvokeRequired)
            {
                FunItem = new EventBus<ItemInfo>.DelegateItemInfoEvent(RefreshItem);
                this.ucList1.Invoke(FunItem, e.Data);
            }
            else
            {
                this.RefreshItem(e.Data);
            }
        }

        private void RefreshItem(List<ItemInfo> item)
        {
            var ls = this.eventBus.GetAll();
            this.ucList1.LoadData(ls);
        }

重點:

1. 捕獲事件:this.eventBus.EventNotice += EventBus_EventNotice;

2. 事件處理方法中,需要判斷是否為UI線程引發,如果不是,則需要委托來進行切換線程,代碼見:private void EventBus_EventNotice(object sender, EventBusArg<List<ItemInfo>> e) 方法

3. 其中FunItem是委托類型的變數,其最終的實現為RefreshItem方法

2.3 A窗體:增加數據

代碼如下:

private EventBus<ItemInfo> eventBus;
        public Form2(EventBus<ItemInfo> eventBus)
        {
            this.eventBus = eventBus;
            InitializeComponent();
        }

        private void button1_Click(object sender, EventArgs e)
        {
            //在UI線程
            this.eventBus.Add(new ItemInfo()
            {
                Title = textBox1.Text,
                Val = Int32.Parse(textBox2.Text)
            });
        }

        private void button2_Click(object sender, EventArgs e)
        {
            //跨線程
            Task.Factory.StartNew(() =>
            {
                for(var i=10; i < 15; i++)
                {
                    this.eventBus.Add(new ItemInfo()
                    {
                        Title = i.ToString() + "-Title",
                        Val = i
                    });
                }
            });
        }

重點:

1. 傳入了EventBus對象的實例,此實例與主界面的EventBus實例為同一個【這點很重要,發佈和訂閱的事件必須在同一實例上】

2. button2_Click事件展示的是跨線程事件,執行此代碼,主界面的刷新會走委托

2.4 B窗體:訂閱列表顯示

代碼如下:

private EventBus<ItemInfo> eventBus;
        public Form3(EventBus<ItemInfo> eventBus)
        {
            this.eventBus = eventBus; 
            InitializeComponent();
            this.eventBus.EventNotice += EventBus_EventNotice;
        }

        private void EventBus_EventNotice(object sender, EventBusArg<List<ItemInfo>> e)
        {
            if (this.ucList1.InvokeRequired)
            {
                var FunItem = new EventBus<ItemInfo>.DelegateItemInfoEvent(RefreshItem);
                this.ucList1.Invoke(FunItem, e.Data);
            }
            else
            {
                this.RefreshItem(e.Data);
            }
        }

        private void RefreshItem(List<ItemInfo> item)
        {
            var ls = this.eventBus.GetAll();
            this.ucList1.LoadData(ls);
        }

        private void Form3_FormClosing(object sender, FormClosingEventArgs e)
        {
            this.eventBus.EventNotice -= EventBus_EventNotice;
        }

重點:

1. 事件的訂閱與取消訂閱,一般情況下可以在關閉窗體時取消訂閱

三、回顧

1. 事件依賴委托,事件可以訂閱和取消訂閱,其訂閱就是為事件增加委托。

2. 委托的本質還是方法(或者說是函數),只不過方法變成了一個變數,可以在運行時動態改變

3. 源碼下載


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

更多相關文章
  • scp是secure copy的簡寫,用於在Linux下進行遠程拷貝文件的命令,和它類似的命令有cp,不過cp只是在本機進行拷貝不能跨伺服器,而且scp傳輸是加密的。可能會稍微影響一下速度。當你伺服器硬碟變為只讀 read only system時,用scp可以幫你把文件移出來。另外,scp還非常不 ...
  • 你可能想創建一個在應用的任何地方都可以訪問的函數,這個教程將幫你實現 👏 很多教程都會說,你在 composer.json 這個文件中通過添加一個自動載入的文件,就可以實現這個需求。但我認為這不是一個好的方式,當你在 helpers.php 文件中添加了更多的函數時,可讀性將變得很差。 下麵我將介 ...
  • 在說正題之前先解釋一下交換機模式是個籠統的稱呼,它不是一個單獨的模式(包括了訂閱模式,路由模式和主題模式),交換機模式是一個比較常用的模式,主要是為了實現數據的同步。 首先,說一下訂閱模式,就和字面上的意思差不多主要就是一個生產者,多個消費者,同一個消息被多個消費者獲取,先看一下官網的圖示 整體執行 ...
  • 當創建隊列jobs、監聽器或訂閱伺服器以推送到隊列中時,您可能會開始認為,一旦分派,隊列工作器決定如何處理您的邏輯就完全由您自己決定了。 嗯……並不是說你不能從作業內部與隊列工作器交互,但是通常情況下,哪怕你做了,也是沒必要的。 這個神奇的騷操作的出現是因為“InteractsWithQueue”這 ...
  • 環境:MacOS 10.13 MAMAP Prophp 7.0.33 + xdebugVisual Studio Code前言我所理解的 POP Chain:利用魔術方法並巧妙構造特殊屬性調用一系列函數或類方法以執行某種敏感操作的調用堆棧反序列化常用魔法函數前言我所理解的 POP Chain:利用魔 ...
  • 【ASP.NET Core學習】在ASP.NET Core 種使用Entity Framework Core介紹,包括如何添加Entity Framwork Core,創建模型和遷移到資料庫,查詢數據,保存數據,使用事務,處理併發衝突 ...
  • [toc] 前言 在之前已經提到過,公用類庫Util已經開源,目的一是為了簡化開發的工作量,畢竟有些常規的功能類庫重覆率還是挺高的,二是為了一起探討學習軟體開發,用的人越多問題也就會越多,解決的問題越多功能也就越完善, 倉庫地址: "April.Util_github" , "April.Util_ ...
  • 在面對對象編程中,類的三大特性分別為封裝,繼承,多態。其中多態的具體實現,依賴於三個方法,也就是虛方法,抽象類和介面。 多態的具體作用是什麼呢?或者說多態的存在有什麼意義呢?多態的存在有效的降低了程式的耦合度,在使用的時候,不僅可以表現大家都有的共性,還能在必要的時候突出一些特殊的的個性。 那麼如何 ...
一周排行
  • 在園子裡面有很多關於各種技術細節的研究文章,都是比較牛逼的框架研究;但是一直沒有看到關於怎麼樣提高開發效率的文章,大多提高開發效率的文章都是關於自動化等方面的輔助工具類型的,而不是開發中的一些小技巧;今天從編碼規範、編碼技巧、開發思想、設計模式等各方面的經驗來分享如何提高開發效率。 ...
  • 前言 隨著近些年微服務的流行,有越來越多的開發者和團隊所採納和使用,它的確提供了很多的優勢也解決了很多的問題,但是我們也知道也並不是銀彈,提供優勢的同時它也給我們的開發人員和團隊也帶來了很多的挑戰。 為了迎接或者採用這些新技術,開發團隊需要更加註重一些流程或工具的使用,這樣才能更好的適應這些新技術所 ...
  • 本文是本系列的完結篇。本系列前面的文章: 邏輯式編程語言極簡實現(使用C#) - 1. 邏輯式編程語言介紹 邏輯式編程語言極簡實現(使用C#) - 2. 一道邏輯題:誰是凶手 邏輯式編程語言極簡實現(使用C#) - 3. 運行原理 下午,吃飽飯的老明和小皮,各拿著一杯剛買的咖啡回到會議室,開始了邏輯 ...
  • 微服務之間的通信之gRPC 介紹 gRPC是一種與語言無關的高性能遠程過程調用 (RPC) 框架,gRPC是Google發佈的基於HTTP 2.0傳輸層協議承載的高性能開源軟體框架,提供了支持多種編程語言的、對網路設備進行配置和納管的方法。由於是開源框架,通信的雙方可以進行二次開發,所以客戶端和服務 ...
  • 一、TLS 線程本地存儲(Thread Local Storage),字面意思就是專屬某個線程的存儲空間。變數大體上分為全局變數和局部變數,一個進程中的所有線程共用地址空間,這個地址空間被劃分為幾個固有的區域,比如堆棧區,全局變數區等,全局變數存儲在全局變數區,虛擬地址固定;局部變數存儲在堆棧區,虛... ...
  • private:私有成員,在類的內部才可以訪問。 protected:保護成員,該類內部和繼承類中可以訪問。 public:公共成員,完全公開,沒有訪問限制。 internal:當前程式集內可以訪問。 ...
  • 前言 上一篇【.Net Core微服務入門全紀錄(六)——EventBus-事件匯流排】中使用CAP完成了一個簡單的Eventbus,實現了服務之間的解耦和非同步調用,並且做到數據的最終一致性。這一篇將使用IdentityServer4來搭建一個鑒權中心,來完成授權認證相關的功能。 IdentitySe ...
  • using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.Linq; using System. ...
  • 從今天起,我將製作一個電影推薦項目,在此寫下博客,記錄每天的成果。 其實,從我發佈 C# 爬取貓眼電影數據 這篇博客後, 我就已經開始製作電影推薦項目了,今天寫下這篇博客,也是因為項目進度已經完成50%了,我就想在這一階段停一下,回顧之前學到的知識。 一、主要為手機端 考慮到項目要有實用性,我選擇了 ...
  • 一、實現Runnable介面 public class RunnableDemo implements Runnable { public void run() { try { Thread.sleep(100); } catch (InterruptedException e) { e.print ...