详解C#之委托

 更新时间:2020年06月04日 11:24:20   作者:千金不如一默  
这篇文章主要介绍了C#委托的含义以及用法,文中代码非常详细,帮助大家更好的理解和学习

委托:顾名思义,让别人帮你办件事。委托是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);
  }
 }

启动控制台:可以看到已经成功把数字打印出来了

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);
  }
 }

以上就是详解C#之委托的详细内容,更多关于C#之委托的资料请关注脚本之家其它相关文章!

相关文章

  • 基于C#后台调用跨域MVC服务及带Cookie验证的实现

    基于C#后台调用跨域MVC服务及带Cookie验证的实现

    本篇文章介绍了,基于C#后台调用跨域MVC服务及带Cookie验证的实现。需要的朋友参考下
    2013-04-04
  • WCF实现的计算器功能实例

    WCF实现的计算器功能实例

    这篇文章主要介绍了WCF实现的计算器功能,结合具体实例形式较为详细的分析了WCF实现计算器功能的具体步骤与相关操作技巧,需要的朋友可以参考下
    2017-06-06
  • 解决安装VS2008无法更改默认路径的问题

    解决安装VS2008无法更改默认路径的问题

    这篇文章主要介绍了安装VS2008无法更改默认路径的解决方法,需要的朋友可以参考下。
    2016-06-06
  • Winform开发框架中如何使用DevExpress的内置图标资源

    Winform开发框架中如何使用DevExpress的内置图标资源

    这篇文章主要给大家介绍了关于在Winform开发框架中如何使用DevExpress的内置图标资源的相关资料,文中通过图文介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们一起来看看吧
    2018-12-12
  • C#异步编程由浅入深(二)之Async/Await的使用

    C#异步编程由浅入深(二)之Async/Await的使用

    这篇文章主要介绍了C#异步编程由浅入深(二)之Async/Await的作用,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-03-03
  • C# OpenVINO读取百度模型实现印章检测

    C# OpenVINO读取百度模型实现印章检测

    这篇文章主要为大家详细介绍了C# OpenVINO如何通过直接读取百度模型实现印章检测,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下
    2023-12-12
  • c#数学表示法(后缀表示法)详解

    c#数学表示法(后缀表示法)详解

    什么是后缀表达式,查了下原来是一种比较特殊的数学表达式,有三种表达式:前缀表达式、中缀表达式和后缀表达式,下面我们使用示例学习一下
    2014-01-01
  • Asp.Net(C#)使用oleDbConnection 连接Excel的方法

    Asp.Net(C#)使用oleDbConnection 连接Excel的方法

    ADO.NET采用不同的Connection对象连接数据库。这篇文章主要介绍了Asp.Net(C#)使用oleDbConnection 连接Excel的方法,非常具有实用价值,需要的朋友可以参考下
    2018-11-11
  • C#中JSON转为实体类和List以及结合使用

    C#中JSON转为实体类和List以及结合使用

    开发中经常遇到将JSON字符串转换为List的需求,下面这篇文章主要给大家介绍了关于C#中JSON转为实体类和List以及结合使用的相关资料,文中通过代码介绍的非常详细,需要的朋友可以参考下
    2024-01-01
  • C# yield在WCF中的错误用法(一)

    C# yield在WCF中的错误用法(一)

    这篇文章主要介绍了C# yield在WCF中的错误使用(一),本文讲解的内容据说是99%的开发人员都有可能犯的错误,需要的朋友可以参考下
    2015-04-04

最新评论