c# 委托的常见用法

  此篇文章是我一个小白对委托的理解和总结,请高手多多评判指教。

  委托就是一种后期绑定机制,说的直白点就是在调用的时候才去传递业务逻辑的一种算法。

委托的创建语法:

public delegate int Comparison<in T>(T left, T right);//官方给出的定义泛型委托的demo

  语法看似像声明一个变量或方法的签名,但实现上是在声明一个类型。编译器会生成一个派生自System.MulticastDelegate的类(而System.MulticastDelegate派生自System.Delegate),类型名与委托的名字相同,其中包含Invoke 、BeginInvoke和EndInvoke等方法。编译器还为这个新类型生成添加和删除处理业务,以便该类的客户端可以在实例的调用列表中添加和删除方法。

  委托可以被定义在类的内部、名称空间下(与类同级)和全局名称空间下(不推荐)。

//在全局定义
public delegate int Comparison<in T>(T left, T right);
namespace Test
{
  //在指定名称空间下定义
  public delegate int Comparison2<in T>(T left, T right);
  public class Student
  {
    //在类内部定义
    public delegate int Comparison3<in T>(T left, T right);
  }
}

委托的定义赋值

  将委托当成类使用(委托本身就是一个类)。

//定义委托
public delegate int Comparison<in T>(T left, T right);
public class Test
{
  //定义
  private Comparison<int> comparator;
  public void Show()
  {
    //赋值 请注意,使用的是方法名称,不带括号,将方法附加给委托作为委托的调用方法。
    this.comparator = Compare;
    //调用
    this.comparator(1, 2);     //调用方式二    this.comparator.Invoke(1, 2);
  }
  private int Compare(int left, int right) => left.CompareTo(right);
}

当用作委托的目标方法是“小方法”的情况下,通常使用lambda表达式语法来执行赋值:

Comparison<string> comparer = (left, right) => left.Length.CompareTo(right.Length);

多播委托

  通常只是将单个目标方法附加到委托。但是,委托对象确实支持将多个目标方法附加到一个委托对象的调用列表,称为多播委托。多播委托意味着通过委托调用时可以调用多个方法,所以可以为委托附加多个方法。

private int Cal(int num) {return num * num;}
private void button1_Click(object sender, EventArgs e)
{
   Func<int, int> action = null;
   //使用+= 或 -= 来增加或移除委托实例
   action += a => { Console.WriteLine($"第{1}次执行" + a); return a + 1; };
   action += Cal;
   action += a => { Console.WriteLine($"第{3}次执行" + a); return a + 3; };
   action -= Cal;
   //如果有返回值的话,返回的是最后一次执行的结果
   int a = action(5);
   Console.WriteLine(a);
}

常用泛型委托

无返回值

public delegate void Action();
public delegate void Action<in T>(T arg);
public delegate void Action<in T1, in T2>(T1 arg1, T2 arg2);
// Other variations removed for brevity.

有返回值

public delegate TResult Func<out TResult>();
public delegate TResult Func<in T1, out TResult>(T1 arg);
public delegate TResult Func<in T1, in T2, out TResult>(T1 arg1, T2 arg2);
// Other variations removed for brevity

返回bool类型

public delegate bool Predicate<in T>(T obj);

注意:在.net core平台中,不支持委托目标方法的异步调用(即不支持BeginInvoke方法):

 Action action = () => Console.WriteLine("委托执行了");
 AsyncCallback asyncCallback = a => Console.WriteLine("回调执行了" + a);
 //System.PlatformNotSupportedException:“Operation is not supported on this platform.”
 IAsyncResult result = action.BeginInvoke(asyncCallback, "asdf"); //委托启动异步调用
 action.EndInvoke(result);

Event 事件

事件是带event关键字的委托的实例,event可以限制被外部调用(invoke)和直接赋值(=)。委托是一个类型,而事件是委托类型的一个实例。

  声明一个事件很简单,只需在声明一个委托对象时加上event关键字就行。

/// <summary>
/// 定义一个委托
/// </summary>
/// <param name="name"></param>
public delegate void ShowInfo(string name);
public class Study
{
  /// <summary>
  /// 声明一个事件
  /// </summary>
  public event ShowInfo ShowInfo;
  public void Show()
  {
    ShowInfo += Study_ShowInfo;
    //只能在“publisher”类中调用
    ShowInfo("asdf");
  }
  private void Study_ShowInfo(string name)
  {
    throw new NotImplementedException();
  }
}

可以在用事件的地方用委托来替代,但事件有一系列规则和约束用以保证程序的安全可控,事件只有 += 和 -= 操作,这样订阅者只能有订阅或取消订阅操作,没有权限执行其它操作。如果是委托,那么订阅者就可以使用 = 来对委托对象重新赋值(其它订阅者全部被取消订阅),甚至将其设置为null,甚至订阅者还可以直接调用委托,这些都是很危险的操作,广播者就失去了独享控制权。

事件保证了程序的安全性和健壮性。

以上就是c# 委托的常见用法的详细内容,更多关于c# 委托的资料请关注我们其它相关文章!

(0)

相关推荐

  • c# 委托的本质是什么

    引言 上一个专题已经和大家分享了我理解的--C#中为什么需要委托,专题中简单介绍了下委托是什么以及委托简单的应用的,在这个专题中将对委托做进一步的介绍的,本专题主要对委本质和委托链进行讨论. 一.委托的本质 平时我们很容易使用委托--用C# delegate关键字定义委托,再用new操作符构造委托实例,然后通过调用委托实例来调用回调方法(就是用一个了委托对象的变量来代替方法名,这句话如果刚接触的人不好理解的话,这里给个例子:MyDelegate mydelegate =new Mydelegat

  • 一篇文章彻底搞清楚c#中的委托与事件

    一.什么是委托呢? 听着名字挺抽象,确实不好理解.面试官最喜欢考察这个,而且更喜欢问:"委托和事件有何异同?".如果对一些知识点没有想明白,那么很容易被绕进去.研究任何事物,我们不妨从它的定义开始,委托也不例外.那么先来看c#中的委托定义,先来个例子: public delegate void GetPacage(string code); 这个委托,看起来就是个方法签名,取包裹,需要验证码.与方法签名不同的地方,在于多了一个delegate.c#中不乏一些便利好用的语法,比如fore

  • 详解c# 委托链

    引言: 上一专题介绍了下编译器是如何来翻译委托的,从中间语言的角度去看委托,希望可以帮助大家进一步的理解委托,然而之前的介绍都是委托只是封装一个方法,那委托能不能封装多个方法呢?因为生活中经常会听到,我代表大家的意见等这样的说话,既然委托也是一个代表,那他如果只能代表一个人,那他的魅力就不是很大了吧,所以我们就会委托能不能代表多个方法的? 答案是可以的,这就是本专题要讲的内容--委托链,委托链也是一个委托,只是因为它是把多个委托链在一起,所以我们就以委托链来这么称呼它的. 一.到底什么是委托链

  • C#委托与匿名委托详解

    本来是想写一篇<委托与lambda表达式的前世今生>,但仅委托部分已经写了很多内容,于是就此分开关于Lambda表达是的内容后续再写吧. 不知道Lambda表达式是谁发明的,只记得第一次接触Lambda表达式是在使用VS2008的时候,那就先认为是微软发明的吧. Lambda表达式从我接触开始到现在变得越来越流行,Java8中开始支持.kotlin更是对C#,F#做了广泛的抄袭(C#曾几何时不也如此对待过Java嘛).其实这都充分说明了,Lambda表达式的重要性.要搞清楚Lambda首先需要

  • C#使用委托的形式调用线程代码实例

    委托 对于委托,我们都知道他是一个引用类型,具有引用类型所具有的通性.需要知道的是它保存的不是实际值,只是是保存对存储在托管堆中的对象的引用.或说的直接点,委托就相当于叫人帮忙,让你帮你做一些事情.我这里就使用委托的形式,调用线程,来简单的说一下. 代码如下: using System; using System.Threading; namespace _012_线程 { class Program { static void Main(string[] args) //在mian中线程是执行

  • C#用委托BeginInvoke做异步线程

    一个应用场景,浏览器上传一个文件,此文件后台调用文件转换,需要耗费相当长的时间,这样,如果是一个线程同步式的做下去,那么用户在浏览器上感觉就是卡住了,卡卡卡卡,这里我们利用委托的BeginInvoke和EndInvoke方法操作线程,BeginInvoke方法可以使用线程异步地执行委托所指向的方法.然后通过EndInvoke方法获得方法的返回值(EndInvoke方法的返回值就是被调用方法的返回值),或是确定方法已经被成功调用,说白了就是相当于开个多线程,你用户文件保存了之后,响应返回,这个Be

  • C#沉淀之委托的深入讲解

    什么是委托 要传递方法,就必须把方法的细节封装在一钟新类型的对象中,即委托.委托是一种特殊类型的对象,其特殊之处在于,我们以前定义的所有对象都包含数据,而委托只包含一个或多个方法的地址. .NET版本中,委托指向方法的地址.在C++中,函数指针是一个指向内存位置的指针,但它不是类型安全的.开发者无法判断这个指针实际指向什么,像参数和返回值等项就更不知道了. .NET委托是类型安全的类,它定义了返回类型和参数的类型.委托类不仅包含对方法的引用,也可以包含对多个方法的引用. 可以认为委托是持有一个或

  • C#匿名委托与Lambda表达式详解

    通过使用匿名委托(匿名方法),使编程变得更加灵活,有关委托与匿名委托请参考我的前一篇Blog<委托与匿名委托>. 继续之前示例,代码如下: static void Main(string[] args) { Worker.TwoNumberHandleMethodDelegate method = delegate(int a, int b) { return a + b; }; Worker worker = new Worker(); int result = worker.HandleT

  • C# 引入委托的目的是什么

    引言: 对于一些刚接触C# 不久的朋友可能会对C#中一些基本特性理解的不是很深,然而这些知识也是面试时面试官经常会问到的问题,所以我觉得有必要和一些接触C#不久的朋友分享下关于C#基础知识的文章,所以有了这个系列,希望通过这个系列让朋友对C#的基础知识理解能够更进一步.然而委托又是C#基础知识中比较重要的一点,基本上后面的特性都和委托有点关系,所以这里就和大家先说说委托,为什么我们需要委托. 一.C#委托是什么的? 在正式介绍委托之前,我想下看看生活中委托的例子--生活中,如果如果我们需要打官司

  • C# WPF 通过委托实现多窗口间的传值的方法

    在使用WPF开发的时候就不免会遇到需要两个窗口间进行传值操作,当然多窗口间传值的方法有很多种,本文介绍的是使用委托实现多窗口间的传值. 在上代码之前呢,先简单介绍一下什么是C#中的委托(如果只想了解如何传值可以略过这部分)在网络上有很多对于委托的介绍和讲解,经过我的学习和总结加上了一点我自己的理解,我认为委托是一种类似于C语言的指针,但是它指向的是方法而不是变量.如果把委托看作一个变量,那么这个变量里存着的就是你目标方法的地址,调用委托约等于调用你的目标方法.(个人理解欢迎指正交流) 以下正文:

  • c# 委托详解

    委托是一个类型.C#中的委托是面向对象的,并且它是类型安全的 当创建委托实例的时候,创建的实例会包含一个调用列表,在调用列表中可以包含多个方法.每个方法称作一个调用实体.调用实体可以是静态方法,也可以是实例方法.如果是实例方法,则该调用实体包含调用该实例方法的实例.委托并不关心它所调用方法所属的类,它只关心被调用方法与委托的类型是否兼容. 下面是代码实例: using System; namespace LycheeTest{ public delegate void D(int a, int

随机推荐