图解C#高级教程(一):委托

ops/2024/10/22 14:04:28/

什么是委托

可以认为委托是持有一个或多个方法的对象。但它与对象不同,因为委托可以被执行。当执行委托时,委托会执行它所“持有”的方法。先看一个完整的使用示例。

// See https://aka.ms/new-console-template for more informationdelegate void MyDel(int value);  // 声明委托类型class Program
{void PrintLow(int value){Console.WriteLine("{0} - Low value", value);}void PrintHigh(int value){Console.WriteLine("{0} - High value", value);}static void Main(string[] args){Program p = new Program();MyDel del;  // 声明委托变量Random rand = new Random();int randomValue = rand.Next(1, 101);del = randomValue < 50 ? p.PrintLow : p.PrintHigh;  // 选择调用哪个方法del(randomValue);  // 调用委托方法}
}

如果生成的随机数 randomValue 小于50,则 del 引用的是 p.PrintLow;否则,del 引用的是 p.PrintHigh

委托概述

下图是使用类和委托的对比:
在这里插入图片描述

可以把 delegate 看成一个包含有序方法列表的对象,这些方法具有相同的签名和返回类型。如下图所示:
在这里插入图片描述

委托的几点说明:

  • 方法的列表称为调用列表;

  • 委托保存的方法可以来自任何类或者结构,只要它们在委托的返回类型和委托的签名(包括ref和out修饰符)上保持一致;

  • 调用列表中的方法可以是实例方法,和静态方法;

  • 执行委托时,会按照方法的添加顺序执行调用列表。

声明委托类型

在这里插入图片描述

创建委托对象

委托是引用类型,因此初始化委托变量需要创建一个对象。有两种方式创建委托对象:

  • 使用 new 运算符:

    class Obj
    {void MyM1(int value) {}static void Other(int value) {}}
    delegate void MyDel(int value);MyDel del1, del2;Obj obj = new Obj();
    del1 = new MyDel(obj.MyM1);  // 使用一个对象的方法
    del2 = new MyDel(Obj.Other);  // 使用静态方法创建委托对象
    
  • 省略 new 运算符:

    MyDel del1, del2;
    del1 = obj.MyM1;  // 使用一个对象的方法
    del2 = Obj.Other;  // 使用静态方法创建委托对象
    

    当为委托变量赋值时,除了为委托分配内存,创建委托对象还会把第一个方法放入委托的调用列表。

    当然,我们还可以在声明委托变量时初始化委托变量。

    MyDel del1 = obj.MyM1;
    

    当我们为同一个委托变量赋值另外一个委托对象时,之前的委托对象就会被垃圾回收器回收。

    MyDel delVar;
    delVar = myInstObj.MyM1;
    ...
    delVar = SClass.OtherM2;
    

在这里插入图片描述

组合委托

可以将两个同类型的委托变量进行 + 操作,赋值给另外一个同类型的新变量,完成委托的组合:

MyDel delA = myInstObj.MyM1;
MyDel delB = SClass.OtherM2;MyDel delC = delA + delB;

在这里插入图片描述

为委托添加或者删除方法

委托可以持有多个方法, 通过使用运算符 +=-= 可以为委托添加或者删除方法。

MyDel del = inst.MyM1;
del += SCl.m3;
del += X.Act;

在这里插入图片描述
从委托移除方法:

del -= SCl.m3;

在这里插入图片描述

移除委托时需要注意以下事项:

  • 如果委托的调用列表中存在多个实例,-= 运算符将从列表的最后开始搜索,并移除第一个与方法匹配的实例;
  • 当要删除的方法在调用列表不存在时,什么也不会发生;
  • 试图调用空委托将会导致异常,因此执行委托之前有必要进行 null 判空。

调用委托

调用委托的方式与调用方法一样。用于调用委托的参数将会传递给调用列表当中的每一个方法(除非有输出参数)。

MyDel delVar = inst.MyM1;
delVar += SCl.m3;
delVar += X.Act;
...
delVar(55);

在这里插入图片描述

下面是一个完整的示例:

delegate void PrintFunction();class Test
{public void Print1(){Console.WriteLine("Print1 -- instance method");}public static void Print2(){Console.WriteLine("Print2 -- static method");}}class Program
{static void Main(){Test t = new Test();PrintFunction pf;
// 实例方法pf = t.Print1;// 给委托增加3个另外的方法pf += Test.Print2;pf += t.Print1;pf += Test.Print2;   // 现在委托含有4个方法if (null != pf){pf();}else{Console.WriteLine("委托为空");}}
}

输出如下:

Print1 -- instance method
Print2 -- static method
Print1 -- instance method
Print2 -- static method

调用带引用参数的委托

如果委托有引用参数,在调用委托列表中的下一个方法时,参数的新值会传给下一个方法。

delegate void MyDel(ref int x);class MyClass
{public void Add2(ref int x) { x += 2; }public void Add3(ref int x) { x += 3; }static void Main(){MyClass mc = new MyClass();MyDel del = mc.Add2;del += mc.Add3;del += mc.Add2;int x = 5;del(ref x);Console.WriteLine("Value: {0}", x);}
}

输出如下:

Value: 12

在这里插入图片描述

Lambda 表达式

C# 当中的 Lambda 表达式是一种简洁的方式来表示匿名方法。通常用于简化代码,尤其是在与 LINQ、委托或事件等相关的场景中。(LINQ 和 事件会在后面相关文章中讲到)。

语法:

(parameters) => expression
  • parameters:代表输入参数。如果只有一个参数,可以省略()
  • =>:称为 lambda 操作符,表示从参数到表达式或代码块的映射。
  • expression:返回的表达式。如果有多条语句,可以使用代码块{}。

Lambda 表达式的常见使用场景

  1. 委托与 Lambda 表达式
delegate void MyDel(int x);MyDel del = x => x *2;

用于 Func<T>委托:

using System;class Program
{static void Main(){// Func 委托,接受两个整数参数,返回它们的和Func<int, int, int> add = (a, b) => a + b;int result = add(3, 4);Console.WriteLine(result);  // 输出 7}
}

Func 是 C# 中一个常用的泛型委托类型,在 C# 3.0 中引入,专门用于表示带有返回值的方法。在使用时,Func<T> 委托的最后一个泛型参数是返回值类型(必须有),前面的泛型参数是输入参数类型(可以没有输入参数)。

  1. Lambda 表达式与 LINQ 查询

    using System;
    using System.Linq;
    using System.Collections.Generic;class Program
    {static void Main(){List<int> numbers = new List<int> { 1, 2, 3, 4, 5, 6 };// 使用 Lambda 表达式筛选出大于 3 的数var result = numbers.Where(n => n > 3).ToList();foreach (var number in result){Console.WriteLine(number);  // 输出 4, 5, 6}}
    }
    
  2. 用于事件处理

    using System;class Program
    {static void Main(){Action<string> messagePrinter = msg => Console.WriteLine(msg);messagePrinter("Hello, Lambda!");  // 输出 "Hello, Lambda!"}
    }

委托的使用场景

委托的主要使用场景如下:

  • 事件处理。
  • 回调函数。

以上两个使用场景的具体例子会在后面文章介绍到。

小结:本章主要介绍了 C# 当中委托和 Lambda 的概念、用法。

各位道友,码字不易,如有收获,记得一键三连啊。


http://www.ppmy.cn/ops/118959.html

相关文章

Python与SQL Server数据库结合导出Excel并做部分修改

Python与SQL Server数据库结合导出Excel并做部分修改 需求&#xff1a;在数据库中提取需要的字段内容&#xff1b;并根据字段内容来提取与拆分数据做为新的列最后导出到Excel文件 # -*- coding: utf-8 -*- import pandas as pd import re import pymssql import timestart_ti…

GAMES101(作业8)

作业8 题目&#xff1a; 模拟绳子动画&#xff0c;包括基于物理的&#xff0c;和非物理的&#xff0c;应该修改的函数是:rope.cpp 中的void Rope::simulateEuler(... Rope::rope(...)&#xff0c;&#xff0c;void Rope::simulateVerlet(...) 代码框架&#xff1a; main:负…

UnityHub下载任意版本的Unity包

1)先打开 // 也可以采用2直接打开 2)也可以直接打开 下载存档 (unity.com) 3)关联起来UnityHub即可

雷池 WAF 如何配置才能正确获取到源 IP

经常有大哥反馈说雷池攻击日志里显示的 IP 有问题。 这里我来讲一下为什么一些情况下雷池显示的攻击 IP 会有问题。 问题说明 默认情况下&#xff0c;雷池会通过 HTTP 连接的 Socket 套接字读取客户端 IP。在雷池作为最外层网管设备的时候这没有问题&#xff0c;雷池获取到的…

Android常用C++特性之std::make_unique

声明&#xff1a;本文内容生成自ChatGPT&#xff0c;目的是为方便大家了解学习作为引用到作者的其他文章中。 std::make_unique 是 C14 引入的一个函数模板&#xff0c;用于创建类型为 std::unique_ptr 的智能指针。智能指针用于管理动态分配的对象&#xff0c;在其生命周期结束…

PHP程序如何实现限制一台电脑登录?

PHP程序如何实现限制一台电脑登录&#xff1f; 可以使用以下几种方法&#xff1a; 1. IP地址限制&#xff1a;在PHP中&#xff0c;可以通过获取客户端的IP地址&#xff0c;然后与允许登录的IP地址列表进行比对。如果客户端的IP地址不在列表中&#xff0c;就禁止登录。 “php $…

Qt 文件操作

目录 Qt 文件操作1. I/O设备1.1 I/O设备的类型1.2 打开模式 2. 文件读写2.1 QFile打开文件写文件读文件静态函数 2.2 StreamQTextStreamQDataStream 2.3 QFileInfo 3. 配置文件3.1 QSettings基本用法设置和获取值配置文件格式常用函数分组操作 3.2 QJsonDocument主要功能解析 J…

ValueError: Out of range float values are not JSON compliant

可能原因一 可能原因二 数据里面有NaN