Tutorialsteacher

关注我们

文章
  • C#
  • C# 面向对象编程
  • ASP.NET Core
  • ASP.NET MVC
  • LINQ
  • 控制反转 (IoC)
  • Web API
  • JavaScript
  • TypeScript
  • jQuery
  • Angular 11
  • Node.js
  • D3.js
  • Sass
  • Python
  • Go 语言
  • HTTPS (SSL)
  • 正则表达式
  • SQL
  • SQL Server
  • PostgreSQL
  • MongoDB
  • C# - 入门
  • C# - 版本历史
  • C# - 第一个程序
  • C# - 关键词
  • C# - 类和对象
  • C# - 命名空间
  • C# - 变量
  • C# - 隐式类型变量
  • C# - 数据类型
  • 数字
  • 字符串
  • DateTime
  • 结构体
  • 枚举
  • StringBuilder
  • 匿名类型
  • 动态类型
  • 可空类型
  • C# - 值类型和引用类型
  • C# - 接口
  • C# - 运算符
  • C# - if else 语句
  • C# - 三元运算符 ?
  • C# - Switch 语句
  • C# - For 循环
  • C# - While 循环
  • C# - Do-while 循环
  • C# - 分部类
  • C# - Static 关键字
  • C# - 数组
  • 多维数组
  • 交错数组
  • C# - 索引器
  • C# - 泛型
  • 泛型约束
  • C# - 集合
  • ArrayList
  • List
  • SortedList
  • Dictionary
  • Hashtable
  • Stack
  • Queue
  • C# - 元组 (Tuple)
  • C# - 值元组 (ValueTuple)
  • C# - 内置异常
  • 异常处理
  • throw 关键字
  • 自定义异常
  • C# - 委托
  • Func 委托
  • Action 委托
  • Predicate 委托
  • 匿名方法
  • C# - 事件
  • C# - 协变
  • C# - 扩展方法
  • C# - 流 I/O
  • C# - File 类
  • C# - FileInfo 类
  • C# - 对象初始化器
  • OOP - 概述
  • 面向对象编程
  • 抽象
  • 封装
  • 关联与组合
  • 继承
  • 多态
  • 方法重写
  • 方法隐藏
  • C# - SOLID 原则
  • 单一职责原则
  • 开闭原则
  • 里氏替换原则
  • 接口隔离原则
  • 依赖倒置原则
  • 设计模式
  • 单例模式
  • 抽象工厂模式
  • 工厂方法模式
Entity Framework Extensions - 提升 EF Core 9
  批量插入
  批量删除
  批量更新
  批量合并

C# - 委托

如果我们想将函数作为参数传递怎么办?C# 如何处理回调函数或事件处理程序?答案是 - 委托。

委托是一种引用类型数据类型,它定义了方法签名。您可以像其他数据类型一样定义委托变量,这些变量可以引用与委托具有相同签名的任何方法。

使用委托涉及三个步骤

  1. 声明委托
  2. 创建实例并引用方法
  3. 调用委托

委托可以使用 delegate 关键字后跟函数签名来声明,如下所示。

委托语法
[access modifier] delegate [return type] [delegate name]([parameters])

以下声明了一个名为 MyDelegate 的委托。

示例:声明委托
public delegate void MyDelegate(string msg);

上面,我们声明了一个名为 MyDelegate 的委托,它具有一个void返回类型和一个字符串参数。委托可以在类外部或类内部声明。实际上,它应该在类外部声明。

声明委托后,我们需要设置目标方法或 lambda 表达式。我们可以通过使用 new 关键字创建委托对象并传入签名与委托签名匹配的方法来完成。

示例:设置委托目标
public delegate void MyDelegate(string msg); // declare a delegate

// set target method
MyDelegate del = new MyDelegate(MethodA);
// or 
MyDelegate del = MethodA; 
// or set lambda expression 
MyDelegate del = (string msg) =>  Console.WriteLine(msg);

// target method
static void MethodA(string message)
{
    Console.WriteLine(message);
}

您可以通过直接赋值方法来设置目标方法,而无需创建委托对象,例如:MyDelegate del = MethodA。

设置目标方法后,可以使用 Invoke() 方法或 () 运算符调用委托。

示例:调用委托
del.Invoke("Hello World!");
// or 
del("Hello World!");

以下是一个完整的委托示例。

示例:委托
public delegate void MyDelegate(string msg); //declaring a delegate

class Program
{
    static void Main(string[] args)
    {
        MyDelegate del = ClassA.MethodA;
        del("Hello World");

        del = ClassB.MethodB;
        del("Hello World");

        del = (string msg) => Console.WriteLine("Called lambda expression: " + msg);
        del("Hello World");
    }
}

class ClassA
{
    static void MethodA(string message)
    {
        Console.WriteLine("Called ClassA.MethodA() with parameter: " + message);
    }
}

class ClassB
{
    static void MethodB(string message)
    {
        Console.WriteLine("Called ClassB.MethodB() with parameter: " + message);
    }
}
尝试一下

下图说明了委托。

C# Delegate

将委托作为参数传递

方法可以具有委托类型的参数,如下所示。

示例:委托
public delegate void MyDelegate(string msg); //declaring a delegate

class Program
{
    static void Main(string[] args)
    {
        MyDelegate del = ClassA.MethodA;
        InvokeDelegate(del);

        del = ClassB.MethodB;
        InvokeDelegate(del);

        del = (string msg) => Console.WriteLine("Called lambda expression: " + msg);
        InvokeDelegate(del);
    }

    static void InvokeDelegate(MyDelegate del) // MyDelegate type parameter
    {
        del("Hello World");
    }
}

class ClassA
{
    static void MethodA(string message)
    {
        Console.WriteLine("Called ClassA.MethodA() with parameter: " + message);
    }
}

class ClassB
{
    static void MethodB(string message)
    {
        Console.WriteLine("Called ClassB.MethodB() with parameter: " + message);
    }
}
尝试一下
注意
在 .NET 中,Func 和 Action 类型是内置的泛型委托,大多数常用委托都应该使用它们,而不是创建新的自定义委托。

多播委托

委托可以指向多个方法。指向多个方法的委托称为多播委托。“+”或“+=”运算符将函数添加到调用列表,而“-”和“-=”运算符将其删除。

示例:多播委托
public delegate void MyDelegate(string msg); //declaring a delegate

class Program
{
    static void Main(string[] args)
    {
        MyDelegate del1 = ClassA.MethodA;
        MyDelegate del2 = ClassB.MethodB;

        MyDelegate del = del1 + del2; // combines del1 + del2
        del("Hello World");

        MyDelegate del3 = (string msg) => Console.WriteLine("Called lambda expression: " + msg);
        del += del3; // combines del1 + del2 + del3
        del("Hello World");

        del = del - del2; // removes del2
        del("Hello World");

        del -= del1 // removes del1
        del("Hello World");
    }
}

class ClassA
{
    static void MethodA(string message)
    {
        Console.WriteLine("Called ClassA.MethodA() with parameter: " + message);
    }
}

class ClassB
{
    static void MethodB(string message)
    {
        Console.WriteLine("Called ClassB.MethodB() with parameter: " + message);
    }
}
尝试一下

加法和减法运算符始终作为赋值的一部分工作:del1 += del2; 完全等同于 del1 = del1+del2;,减法同理。

如果委托返回一个值,则当调用多播委托时,将返回最后分配的目标方法的值。

示例:返回值的多播委托
public delegate int MyDelegate(); //declaring a delegate

class Program
{
    static void Main(string[] args)
    {
        MyDelegate del1 = ClassA.MethodA;
        MyDelegate del2 = ClassB.MethodB;

        MyDelegate del = del1 + del2; 
        Console.WriteLine(del());// returns 200
    }
}

class ClassA
{
    static int MethodA()
    {
        return 100;
    }
}

class ClassB
{
    static int MethodB()
    {
        return 200;
    }
}
尝试一下

泛型委托

泛型委托的定义方式与委托相同,但使用泛型类型参数或返回类型。设置目标方法时必须指定泛型类型。

例如,考虑以下用于 int 和 string 参数的泛型委托。

示例:泛型委托
public delegate T add<T>(T param1, T param2); // generic delegate

class Program
{
    static void Main(string[] args)
    {
        add<int> sum = Sum;
        Console.WriteLine(sum(10, 20));

        add<string> con = Concat;
        Console.WriteLine(conct("Hello ","World!!"));
    }

    public static int Sum(int val1, int val2)
    {
        return val1 + val2;
    }

    public static string Concat(string str1, string str2)
    {
        return str1 + str2;
    }
}
尝试一下

委托也用于声明事件和匿名方法。

在此处了解有关委托的更多信息。

要记住的要点
  1. 委托是定义签名的引用类型数据类型。
  2. 委托类型变量可以引用与委托具有相同签名的任何方法。
  3. 语法:[访问修饰符] delegate [返回类型] [委托名称]([参数])
  4. 目标方法的签名必须与委托签名匹配。
  5. 委托可以像普通函数或 Invoke() 方法一样调用。
  6. 可以使用“+”或“+=”运算符将多个方法分配给委托,并使用“-”或“-=”运算符删除。这称为多播委托。
  7. 如果多播委托返回一个值,则它返回最后分配的目标方法的值。
  8. 委托用于在 C# 中声明事件和匿名方法。
TUTORIALSTEACHER.COM

TutorialsTeacher.com 是您权威的技术教程来源,旨在通过循序渐进的方法,指导您掌握各种网络和其他技术。

我们的内容可帮助所有级别的学习者轻松快速地学习技术。通过访问此平台,您承认您已阅读并同意遵守我们的使用条款和隐私政策,这些条款和政策旨在保护您的体验和隐私权。

[email protected]

关于我们使用条款隐私政策
copywrite-symbol

2024 TutorialsTeacher.com. (v 1.2) 版权所有。