C#之委托

来源:https://www.cnblogs.com/jixiaosa/archive/2019/04/11/10687068.html
-Advertisement-
Play Games

委托:顧名思義,讓別人幫你辦件事。委托是C#實現回調函數的一種機制。可能有人會問了,回調函數是個啥??? 舉個例子:我現在是一家公司的老闆,公司現在在招聘.NET工程師,我們有一個小姐姐專門負責接受求職者投遞的簡歷,我就告訴這個小姐姐,一旦收到新的簡歷就轉發給我一份。 這個例子里小姐姐要做的工作:給 ...


委托:顧名思義,讓別人幫你辦件事。委托是C#實現回調函數的一種機制。可能有人會問了,回調函數是個啥???

舉個例子:我現在是一家公司的老闆,公司現在在招聘.NET工程師,我們有一個小姐姐專門負責接受求職者投遞的簡歷,我就告訴這個小姐姐,一旦收到新的簡歷就轉發給我一份。

這個例子里小姐姐要做的工作:給我轉發一份簡歷(回調函數里的操作),就是一個回調函數的作用。一旦有了滿足條件(收到了新的簡歷),小姐姐就會轉發給我(觸發回調函數

用來代碼來看看是怎麼實現的:

1.定義一個委托:

   // 定義委托,這個委托需要獲取一個int型參數,返回void
        internal delegate void Feedback(int value);

 

2.定義回調方法:這裡定義了兩個方法,一個靜態,一個實例。正好看看調用方式的不同。註意:定義的回調方法簽名必須和委托對象一致(這裡都是int 類型參數,沒有返回值。這麼說也不全對,涉及到協變逆變。這裡就不解釋這倆了),這是因為將方法綁定到委托時,編譯器會檢測他們的相容性。不符合的話回報編譯錯誤。就比如有一個方法要傳入String類型,我們給它傳遞了一個int類型一樣。

這裡為了方便演示就只把數字列印在了控制台。

 /// <summary>
        /// 靜態回調方法
        /// </summary>
        /// <param name="value"></param>
        private static void FeedbackToConsole(int value)
        {
            // 依次列印數字
            Console.WriteLine("Item=" + value);
        }
        /// <summary>
        /// 實例回調方法
        /// </summary>
        /// <param name="value"></param>
        private void InstanceFeedbackToConsole(int value)
        {
            Console.WriteLine("Item=" + value);
        }

 

3.編寫一個方法來觸發回調函數:有三個參數:前兩個做迴圈使用,後一個接收定義的委托對象。內部代碼迴圈調用回調方法 fb(val)的寫法,其實就是相當於要調用的函數。例:

FeedbackToConsole(val)
        /// <summary>
        /// 使用此方法觸發委托回調
        /// </summary>
        /// <param name="from">開始</param>
        /// <param name="to">結束</param>
        /// <param name="fb">委托引用</param>
        private static void Counter(int from,int to, Feedback fb)
        {
            for (int val = from; val <= to; val++)
            {
                // fb不為空,則調用回調方法
                if (fb != null)
                {
                    fb(val);
                }
                //fb?.Invoke(val); 簡化版本調用
            }
        }

4.定義Counter的方法調用(這一步可有可無,為了區分靜態和實例方法就寫了)

第一次調用Counter,傳遞Null,在回調方法里有一步判空操作,所以是不回調用回調函數的。第二個Counter調用正常傳遞參數,構造一個委托對象並綁定了一個方法

/// <summary>
        /// 靜態調用
        /// </summary>
        private static void StaticDelegateDemo()
        {
            Console.WriteLine("---------委托調用靜態方法------------");
            Counter(1, 10, null);
            Counter(1, 10, new Feedback(FeedbackToConsole));
            
            

        }

        /// <summary>
        /// 實例調用
        /// </summary>
        private static void InstanceDelegateDemo()
        {
            Console.WriteLine("---------委托調用實例方法------------");
            Program p = new Program();
            Counter(1, 10, null);
            Counter(1, 5, new Feedback(p.InstanceFeedbackToConsole));
        }

 5. 查看控制台信息

完整代碼:

 class Program
    {
        // 定義委托,並引用一個方法,這個方法需要獲取一個int型參數返回void
        internal delegate void Feedback(int value);
        static void Main(string[] args)
        {
          
            StaticDelegateDemo();
            InstanceDelegateDemo();
            Console.ReadKey();
        }
        
        /// <summary>
        /// 靜態調用
        /// </summary>
        private static void StaticDelegateDemo()
        {
            Console.WriteLine("---------委托調用靜態方法------------");
            Counter(1, 10, null);
            Counter(1, 10, new Feedback(FeedbackToConsole));
           

        }

        /// <summary>
        /// 實例調用
        /// </summary>
        private static void InstanceDelegateDemo()
        {
            Console.WriteLine("---------委托調用實例方法------------");
            Program p = new Program();
            Counter(1, 10, null);
            Counter(1, 5, new Feedback(p.InstanceFeedbackToConsole));
        }


        /// <summary>
        /// 靜態回調方法
        /// </summary>
        /// <param name="value"></param>
        private static void FeedbackToConsole(int value)
        {
            // 依次列印數字
            Console.WriteLine("Item=" + value);
        }
        /// <summary>
        /// 實例回調方法
        /// </summary>
        /// <param name="value"></param>
        private void InstanceFeedbackToConsole(int value)
        {
            Console.WriteLine("Item=" + value);
        }
    }
View Code

 

啟動控制台:可以看到已經成功把數字列印出來了

6. 委托鏈:委托鏈是委托對象的集合。可以利用委托鏈調用集合中的委托所綁定的全部方法。繼續在原有的基礎上添加委托鏈的方法。

新添加的兩個方法本質上沒有區別都是對委托鏈的實現,不同的是寫法,明顯是第二個方法更加精簡一些。這是因為C#編譯器重載了+=和-=操作符,這兩個操作符分別調用Combine和Remove。

/// <summary>
        /// 委托鏈調用 1
        /// </summary>
        /// <param name="p"></param>
        private static void ChainDelegateDemo(Program p)
        {
            Console.WriteLine("---------委托鏈調用1------------");
            Feedback fb1 = new Feedback(FeedbackToConsole);
            Feedback fb2 = new Feedback(p.InstanceFeedbackToConsole);
            Feedback fbChain = null;
            fbChain = (Feedback)Delegate.Combine(fbChain, fb1);
            fbChain = (Feedback)Delegate.Combine(fbChain, fb2);
            Counter(1, 3, fbChain);
            Console.WriteLine();
            fbChain = (Feedback)Delegate.Remove(fbChain, new Feedback(FeedbackToConsole));
            Counter(1, 3, fbChain);
        }

        /// <summary>
        /// 委托鏈調用 2
        /// </summary>
        /// <param name="p"></param>
        private  static void ChainDelegateDemo2(Program p)
        {
            Console.WriteLine("---------委托鏈調用2------------");
            Feedback fb1 = new Feedback(FeedbackToConsole);
            Feedback fb2 = new Feedback(p.InstanceFeedbackToConsole);
            Feedback fbChain = null;
            fbChain += fb1;
            fbChain += fb2;
            Counter(1, 3, fbChain);
            Console.WriteLine();
            fbChain -= new Feedback(FeedbackToConsole);
            Counter(1, 2, fbChain);
        }

在Main方法添加對委托鏈的調用:

 static void Main(string[] args)
        {
            Program p = new Program();
            StaticDelegateDemo();
            InstanceDelegateDemo();
            ChainDelegateDemo(p);
            ChainDelegateDemo2(p);
            Console.WriteLine("Hello World!");
            Console.ReadKey();
        }

啟動項目:

7. C#為委托提供的簡化:

7.1 不需要構造委托對象:

之前的代碼:

Counter(1, 10, new Feedback(FeedbackToConsole));

構造了一個委托對象並傳遞給Counter方法,由於C#編譯器能自己推斷。所以可以省略構造委托對象,直接傳遞方法。使代碼的可讀性更佳,也更容易理解。

簡化後的代碼:

        /// <summary>
        /// 靜態調用
        /// </summary>
        private static void StaticDelegateDemo()
        {
            Console.WriteLine("---------委托調用靜態方法------------");
            Counter(1, 10, null);
            //Counter(1, 10, new Feedback(FeedbackToConsole));
            Counter(1, 10, FeedbackToConsole);
            

        }    

可以看到效果是一樣的:

7.2 簡化語法:不需要定義回調方法(以lambda表達式實現)

在前面的代碼中定義了一個回調方法:

        /// <summary>
        /// 靜態回調方法
        /// </summary>
        /// <param name="value"></param>
        private static void FeedbackToConsole(int value)
        {
            // 依次列印數字
            Console.WriteLine("Item=" + value);
        }

現在以lambda表達式方式實現:

        /// <summary>
        /// 靜態調用
        /// </summary>
        private static void StaticDelegateDemo()
        {
            Console.WriteLine("---------委托調用靜態方法------------");
            Counter(1, 10, null);
            //Counter(1, 10, new Feedback(FeedbackToConsole));
            //Counter(1, 10, FeedbackToConsole);
            Counter(1, 10, value => Console.WriteLine(value));

        }

lambda表達式實際上是一個匿名函數。編譯器在看到lambda之後會在類中自動定義一個新的私有方法。類似於之前寫的回調方法FeedbackToConsole()。lambda必須匹配委托!

lambda的語法: 參數 => 返回值。

=>左邊是要傳入的參數,本例中是傳入一個Int類型的變數,=>右邊是具體的代碼,相當於FeedbackToConsole(),{}中所做的操作

一些規則:

如果不傳遞參數: ()=>Console.WriteLine("Hello World!")

傳遞一個參數:(int n)=>Console.WriteLine(n.ToString())    或者去掉()和int  編譯器會自己推斷類型:n=>Console.WriteLine(n.ToString())

傳遞多個參數:(int n ,int m)=>Console.WriteLine(n.ToString())  或者編譯器自己推斷類型:(n , m)=>Console.WriteLine(n.ToString())

註:如果有一個方法需要多處調用或者方法裡面的代碼量較多。還是單獨寫一個方法較為理想。

最後看一下換成lambda的寫法結果顯示是否一樣

 

全部代碼:

    class Program
    {
        // 定義委托,並引用一個方法,這個方法需要獲取一個int型參數返回void
        internal delegate void Feedback(int value);
        static void Main(string[] args)
        {
            Program p = new Program();
            StaticDelegateDemo();
            InstanceDelegateDemo();
            ChainDelegateDemo(p);
            ChainDelegateDemo2(p);
            Console.WriteLine("Hello World!");
            string[] names = { "Jeff", "Jee", "aa", "bb" };
            //char find = 'e';
            //names= Array.FindAll(names, name => name.IndexOf(find) >= 0);
            //Array.ForEach(names, Console.WriteLine);
            Console.ReadKey();
        }
        
        /// <summary>
        /// 靜態調用
        /// </summary>
        private static void StaticDelegateDemo()
        {
            Console.WriteLine("---------委托調用靜態方法------------");
            Counter(1, 10, null);
            //Counter(1, 10, new Feedback(FeedbackToConsole));
            //Counter(1, 10, FeedbackToConsole);
            Counter(1, 10, value => Console.WriteLine(value));

        }

        /// <summary>
        /// 實例調用
        /// </summary>
        private static void InstanceDelegateDemo()
        {
            Console.WriteLine("---------委托調用實例方法------------");
            Program p = new Program();
            Counter(1, 10, null);
            Counter(1, 5, new Feedback(p.InstanceFeedbackToConsole));
        }

        /// <summary>
        /// 委托鏈調用 1
        /// </summary>
        /// <param name="p"></param>
        private static void ChainDelegateDemo(Program p)
        {
            Console.WriteLine("---------委托鏈調用1------------");
            Feedback fb1 = new Feedback(FeedbackToConsole);
            Feedback fb2 = new Feedback(p.InstanceFeedbackToConsole);
            Feedback fbChain = null;
            fbChain = (Feedback)Delegate.Combine(fbChain, fb1);
            fbChain = (Feedback)Delegate.Combine(fbChain, fb2);
            Counter(1, 3, fbChain);
            Console.WriteLine();
            fbChain = (Feedback)Delegate.Remove(fbChain, new Feedback(FeedbackToConsole));
            Counter(1, 3, fbChain);
        }

        /// <summary>
        /// 委托鏈調用 2
        /// </summary>
        /// <param name="p"></param>
        private  static void ChainDelegateDemo2(Program p)
        {
            Console.WriteLine("---------委托鏈調用2------------");
            Feedback fb1 = new Feedback(FeedbackToConsole);
            Feedback fb2 = new Feedback(p.InstanceFeedbackToConsole);
            Feedback fbChain = null;
            fbChain += fb1;
            fbChain += fb2;
            Counter(1, 3, fbChain);
            Console.WriteLine();
            fbChain -= new Feedback(FeedbackToConsole);
            Counter(1, 2, fbChain);
        }
        /// <summary>
        /// 使用此方法觸發委托回調
        /// </summary>
        /// <param name="from">開始</param>
        /// <param name="to">結束</param>
        /// <param name="fb">委托引用</param>
        private static void Counter(int from,int to, Feedback fb)
        {
            for (int val = from; val <= to; val++)
            {
                // fb不為空,則調用回調方法
                if (fb != null)
                {
                    fb(val);
                }
                //fb?.Invoke(val); 簡化版本調用
            }
        }

        /// <summary>
        /// 靜態回調方法
        /// </summary>
        /// <param name="value"></param>
        private static void FeedbackToConsole(int value)
        {
            // 依次列印數字
            Console.WriteLine("Item=" + value);
        }
        /// <summary>
        /// 實例回調方法
        /// </summary>
        /// <param name="value"></param>
        private void InstanceFeedbackToConsole(int value)
        {
            Console.WriteLine("Item=" + value);
        }
    }
View Code

 

github:

https://github.com/xiaoMaPrincess/CSharp


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

-Advertisement-
Play Games
更多相關文章
  • 上一篇講到方法的調用和簡單的構造方法,今天繼續加深,加參數或者該參數; package sklx; public class Car{ //設三個屬性 private String 品牌; private int 價格; private String 顏色; //修改屬性參數方法 public Ca ...
  • 我們面試中經常會被問到多線程相關知識,這一塊內容往淺了說大家都會,但是一問到底層實現原理,我們往往就一臉懵逼。 這段時間準備好好學習多線程,接下來會寫一系列關於多線程的知識。 我們首先要瞭解線程,百度百科這麼介紹:線程(thread)是操作系統能夠進行運算調度的最小單位。它被包含在進程之中,是進程中 ...
  • 思路 **先考慮一條鏈的情況怎麼做。** 因為只有兩個子樹,並且兩個子樹都是鏈。所以可以把這兩條鏈找出來,然後$sort$一下。合併起來。 **然後推廣到樹上** 對於每一棵樹都可以按照和上面同樣的方法合併成一條鏈。 ...
  • 一. 概述 本篇開始進入IS4實戰學習,從第一個示例開始,該示例是 “使用客戶端憑據保護API”,這是使用IdentityServer保護api的最基本場景。該示例涉及到三個項目包括:IdentityServer項目、API項目、Client項目,都有自己的宿主,為了方便開發,放在了一個解決方案下( ...
  • 導入導出的方法以及引用,可以自行創建一個幫助類 using System;using NPOI.SS.UserModel;using NPOI.XSSF.UserModel;using NPOI.HSSF.UserModel;using System.IO;using System.Data;usi ...
  • 背景 1. 基於之前 "基於Log4Net本地日誌服務簡單實現" 實現本地日誌服務,但是隨著項目開發演進,本地日誌服務滿足不了需求,譬如在預發佈環境或者生產環境,不可能讓開發人員登錄查看本地日誌文件分析。 2. Kafka+ELK日誌服務套件,可以線上日誌服務可以解決上述問題,並且提供豐富報表分析等 ...
  • 1. 下載windbg並安裝。 我下載的是 Windbg 6.12。註意,windbg分32位和64位,由分析環境的位數決定。我這裡安裝的是32位的。安裝過程很簡單,一路next就可以。 2. 準備被調試的程式。 新建一個C#控制台程式,使用如下代碼。編譯~ class Program { stat ...
  • 我們在用EF從資料庫生成模型的時候,預設實體類是沒有註釋的,但是我們已經在資料庫欄位添加說明瞭,能不能自動把註釋也拿過來? 答案是:能。 那麼我們開始 首先隨便開一個ASP.NET MVC項目,我們添加ADO實體數據模型。添加完成後我們打開userinfo.cs(這裡我的模型名稱為userinfo) ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...