.NET Core 委托原理剖析

打印 上一主题 下一主题

主题 865|帖子 865|积分 2595

.NET Core 委托原理剖析

在 .NET Core 中,委托(Delegate)是一种类型安全的函数指针,它允许你将方法作为参数通报给其他方法,或者将方法存储在变量中以便稍后调用。委托在事件处理、回调机制以及异步编程中非常有用。理解委托的运行原理对于把握 .NET Core 的高级编程本领至关重要。
1. 委托的基本概念

委托是一种引用类型,它引用一个或多个方法。委托定义了方法的署名(参数类型和返回类型),因此只有具有雷同署名的方法才气被委托引用。
1.1 定义委托

你可以通过 delegate 关键字来定义一个委托类型。例如:
  1.  // 定义一个委托类型
  2. public delegate void GreetDelegate(string name);
复制代码
这个委托类型 GreetDelegate 可以引用任何具有 void 返回类型和 string 参数的方法。
1.2 实例化委托

一旦定义了委托类型,你可以创建该委托的实例,并将方法赋值给它。例如:
  1.   // 与委托签名匹配的方法
  2.   public void Greet(string name)
  3.   {
  4.       Console.WriteLine($"Hello, {name}!");
  5.   }
  6.   // 创建委托实例并绑定方法
  7.   GreetDelegate del  = new GreetDelegate(Greet);
复制代码
在这个例子中,del 是一个委托实例,它引用了 Greet 方法。
1.3 调用委托

你可以像调用方法一样调用委托:
  1. // 调用委托
  2. del("World");
复制代码
这会调用 del 方法,并输出 "Hello, World!"。
2.委托的类型

1. 单播委托(Singlecast Delegate)

单播委托是指一个委托实例只能引用一个方法。这是最基本的委托类型。
  1. public delegate void GreetDelegate(string message);
  2. public void ShowMessage(string message)
  3. {
  4.     Console.WriteLine(message);
  5. }
  6. GreetDelegate del = new GreetDelegate(ShowMessage);
  7. del("Hello, World!"); // 输出:Hello, World!
复制代码
2. 多播委托(Multicast Delegate)

多播委托是指一个委托实例可以引用多个方法。通过 += 运算符可以将多个方法添加到委托实例中,并通过 -= 运算符移除方法。
  1. public void ShowMessage1(string message)
  2. {
  3.     Console.WriteLine($"Message 1: {message}");
  4. }
  5. public void ShowMessage2(string message)
  6. {
  7.     Console.WriteLine($"Message 2: {message}");
  8. }
  9. public void ShowMessage3(string message)
  10. {
  11.     Console.WriteLine($"Message 3: {message}");
  12. }
  13. GreetMulticastDelegate del = new GreetMulticastDelegate(ShowMessage1);
  14. del += ShowMessage2;
  15. del += ShowMessage3;
  16. del -= ShowMessage2;
  17. del("Hello, World!");
复制代码
在这个例子中,del 委托实例引用了多个方法:将ShowMessage1,ShowMessage2 和 ShowMessage3添加到了多播委托实例中,然后并通过 -= 运算符移除ShowMessage2。然后调用 del("Hello, World!") 时,三个方法都会被调用,输出如下:
  1. Message 1: Hello, World!
  2. //ShowMessage2方法已移除
  3. Message 3: Hello, World!
复制代码
3. 泛型委托(Generic Delegate)

泛型委托是 C# 中的一种特殊委托类型,它允许你定义可以处理多种数据类型的委托。通过使用泛型,你可以编写更通用、更灵活的代码,而不需要为每种数据类型单独定义委托。
以下是几个泛型委托的示例,展示了如何使用泛型委托处理不同类型的数据。
3.1 简单的泛型委托
  1. public delegate T MyGenericDelegate<T>(T arg);
  2. public int Square(int x)
  3. {
  4.     return x * x;
  5. }
  6. public string Reverse(string s)
  7. {
  8.     return new string(s.Reverse().ToArray());
  9. }
  10. MyGenericDelegate<int> intDelegate = new MyGenericDelegate<int>(Square);
  11. Console.WriteLine(intDelegate(5)); // 输出:25
  12. MyGenericDelegate<string> stringDelegate = new MyGenericDelegate<string>(Reverse);
  13. Console.WriteLine(stringDelegate("hello")); // 输出:olleh
复制代码

  • 说明

    • • MyGenericDelegate 实例化了一个处理 int 类型数据的委托。
    • • MyGenericDelegate 实例化了一个处理 string 类型数据的委托。

3.2 多参数泛型委托
  1. public delegate TResult MyGenericDelegate<T1, T2, TResult>(T1 arg1, T2 arg2);
  2. public int Add(int a, int b)
  3. {
  4.     return a + b;
  5. }
  6. public string Concat(string s1, string s2)
  7. {
  8.     return s1 + s2;
  9. }
  10. MyGenericDelegate<int, int, int> intDelegate = new MyGenericDelegate<int, int, int>(Add);
  11. Console.WriteLine(intDelegate(3, 5)); // 输出:8
  12. MyGenericDelegate<string, string, string> stringDelegate = new MyGenericDelegate<string, string, string>(Concat);
  13. Console.WriteLine(stringDelegate("Hello, ", "World!")); // 输出:Hello, World!
复制代码

  • 说明

    • • MyGenericDelegate 实例化了一个处理两个 int 类型参数并返回 int 类型结果的委托。
    • • MyGenericDelegate 实例化了一个处理两个 string 类型参数并返回 string 类型结果的委托。

4. 内置委托类型

C# 提供了一些内置的泛型委托类型,可以直接使用,而无需自定义委托。
4.1 Action 委托

Action 委托用于引用没有返回值的方法。它可以有 0 到 16 个参数。
  1. Action<string> action = (message) => Console.WriteLine(message);
  2. action("Hello, World!"); // 输出:Hello, World!
复制代码
4.2 Func 委托

Func 委托用于引用有返回值的方法。它可以有 0 到 16 个参数,最后一个泛型参数是返回值类型。
  1. Func<int, int, int> add = (a, b) => a + b;
  2. Console.WriteLine(add(3, 5)); // 输出:8
复制代码
4.3 Predicate 委托

Predicate 委托用于引用返回布尔值的方法,通常用于条件判断。
  1. Predicate<int> isEven = (num) => num % 2 == 0;
  2. Console.WriteLine(isEven(4)); // 输出:True
复制代码
5. 匿名方法委托

匿名方法允许你直接定义委托的实现,而无需显式声明一个方法。
  1. MyDelegate del = delegate(string message)
  2. {
  3.     Console.WriteLine(message);
  4. };
  5. del("Hello, World!"); // 输出:Hello, World!
复制代码
6. Lambda 表达式委托

Lambda 表达式是一种更轻便的匿名方法写法,通常用于定义委托。
  1. Action<string> action = (message) => Console.WriteLine(message);
  2. action("Hello, World!"); // 输出:Hello, World!Func<int, int, int> add = (a, b) => a + b;
  3. Console.WriteLine(add(3, 5)); // 输出:8
复制代码
7. 事件委托

事件是一种特殊的委托,通常用于实现观察者模式。事件委托通常与 EventHandler 或 EventHandler 一起使用。
  1. public class Button
  2. {
  3.     public event EventHandler Click;
  4.     public void OnClick()
  5.     {
  6.         Click?.Invoke(this, EventArgs.Empty);
  7.     }
  8. }
  9. public class Program
  10. {
  11.     public static void Main()
  12.     {
  13.         Button button = new Button();
  14.         button.Click += (sender, e) => Console.WriteLine("Button clicked!");
  15.         button.OnClick(); // 输出:Button clicked!
  16.     }
  17. }
复制代码
8. 异步委托

异步委托允许你异步调用方法,通常与 BeginInvoke 和 EndInvoke 一起使用。
  1. public delegate int MyAsyncDelegate(int x, int y);
  2. public int Add(int a, int b)
  3. {
  4.     return a + b;
  5. }
  6. MyAsyncDelegate del = new MyAsyncDelegate(Add);
  7. IAsyncResult result = del.BeginInvoke(3, 5, null, null);
  8. int sum = del.EndInvoke(result);
  9. Console.WriteLine(sum); // 输出:8
复制代码
9. 动态委托

动态委托允许你在运行时动态创建和调用委托。
  1. public int Multiply(int a, int b)
  2. {
  3.     return a * b;
  4. }
  5. var method = typeof(Program).GetMethod("Multiply");
  6. var del = Delegate.CreateDelegate(typeof(Func<int, int, int>), null, method);
  7. int result = (del as Func<int, int, int>)(3, 5);
  8. Console.WriteLine(result); // 输出:15
复制代码
3. 委托的运行原理

委托的运行原理涉及到 .NET Core 的运行机遇制和内部实现。以下是委托运行原理的关键点:
3.1 委托的内部布局

在 .NET Core 中,委托是一个类,它继承自 System.MulticastDelegate 类。System.MulticastDelegate 类又继承自 System.Delegate 类。委托类包含以下关键成员:

  • • _target:指向调用方法的对象实例(如果是静态方法,则为 null)。
  • • _methodPtr:指向方法的函数指针。
  • • _invocationList:用于存储多播委托中的多个方法。
3.2 委托的调用

当你调用委托时,.NET Core 运行时会实行以下步骤:

  • 1. 查抄委托实例是否为**** **null**:如果委托实例为 null,则会抛出 NullReferenceException。
  • 2. 调用委托的**** **Invoke** ****方法:委托实例的 Invoke 方法会被调用,该方法会根据 _target 和 _methodPtr 调用现实的方法。
  • 3. 处理多播委托:如果委托是多播委托(即 _invocationList 不为 null),则 Invoke 方法会遍历 _invocationList,依次调用每个方法。
3.3 委托的优化

.NET Core 对委托的调用进行了优化,以提高性能。例如,对于单播委托(即只引用一个方法的委托),.NET Core 会直接调用方法,而不需要通过 Invoke 方法。
4. 委托的应用场景

委托在 .NET Core 中有多种应用场景,以下是一些常见的场景:
4.1 事件处理

委托在事件处理中非常常见。事件是一种特殊的委托,它允许对象在发生某些事变时通知其他对象。例如:
  1. C#public class Button
  2. {
  3.     public event Action Click;
  4.     public void OnClick()
  5.     {
  6.         Click?.Invoke();
  7.     }
  8. }
  9. public class Program
  10. {
  11.     public static void Main()
  12.     {
  13.         Button button = new Button();
  14.         button.Click += () => Console.WriteLine("Button clicked!");
  15.         button.OnClick();
  16.     }
  17. }
复制代码
在这个例子中,Button 类定义了一个 Click 事件,当 OnClick 方法被调用时,事件处理程序会被触发。
4.2 回调机制

委托可以用于实现回调机制,允许一个方法在完成时通知另一个方法。例如:
  1. public void DoWork(Action callback)
  2. {
  3.     // 执行一些工作
  4.     Console.WriteLine("Work is done.");
  5.     // 调用回调方法
  6.     callback?.Invoke();
  7. }
  8. public void Main()
  9. {
  10.     DoWork(() => Console.WriteLine("Callback called."));
  11. }
复制代码
在这个例子中,DoWork 方法在完成工作后调用传入的回调方法。
4.3 异步编程

委托在异步编程中也非常有用。例如,Task 类的 ContinueWith 方法允许你在使命完成时实行一个委托:
  1. Task.Run(() => Console.WriteLine("Task is running..."))
  2.     .ContinueWith(task => Console.WriteLine("Task is completed."));
复制代码
5. 总结

委托是 .NET Core 中一个非常强大的特性,它允许你将方法作为参数通报、存储和调用。理解委托的运行原理有助于你更好地使用这一特性,特殊是在事件处理、回调机制和异步编程中。通过把握委托,你可以编写更加灵活和可扩展的代码。


免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

欢乐狗

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表