C# 委托中 Invoke/BeginInvoke/EndInvoke和DynamicInvoke 方法

devtools/2025/3/13 4:56:42/

总目录


前言

在C#中,委托(Delegate)提供了多种调用方式,包括 InvokeBeginInvokeEndInvokeDynamicInvoke。每种调用方式都有其特定的用途和适用场景。下面将详细介绍这些方法的区别与联系。


一、 Invoke方法

1. 定义

Invoke 是同步调用委托的方法。它会阻塞当前线程,直到委托所引用的方法执行完毕并返回结果。

2. 特点

  • 同步调用:当前线程会被阻塞,直到委托方法执行完成。
  • 简单直接:适用于不需要异步处理的场景。

3. 示例代码

public class Program
{public static void Main(){// 定义委托Action action = () => Console.WriteLine("Hello, World!");// 同步调用action.Invoke();}
}
public class Program
{public delegate string SimpleDelegate(string message);public static string PrintMessage(string message){Console.WriteLine($"Message: {message}");return "Done";}public static void Main(){SimpleDelegate del = PrintMessage;string result = del.Invoke("Hello, World!"); // 同步调用Console.WriteLine(result);}
}

输出结果

Message: Hello, World!
Done

二、 BeginInvokeEndInvoke 方法

1. 定义

BeginInvokeEndInvoke 方法用于异步调用委托所引用的方法。BeginInvoke 方法启动异步操作并立即返回一个 IAsyncResult 对象,该对象可以用于跟踪异步操作的状态。而 EndInvoke 方法用于获取异步调用的结果或等待异步调用完成。

2. 特点

  • BeginInvoke
    • 异步调用:当前线程不会被阻塞,委托方法将在后台线程上执行。
    • 回调机制:可以通过提供一个回调函数,在委托方法完成后自动调用该回调函数。
    • 参数传递:除了委托方法的参数外,还需要传递一个 AsyncCallback 委托和一个用户定义的对象(通常是 null)。
  • EndInvoke
    • 获取结果:通过传入 BeginInvoke 返回的 IAsyncResult 对象来获取异步调用的结果。
    • 等待完成:如果异步调用尚未完成,EndInvoke 将阻塞当前线程,直到异步调用完成。

3. 示例代码

using System;public class Program
{public delegate string SimpleDelegate(string message);public static string PrintMessage(string message){Console.WriteLine($"Message: {message}");return "Done";}public static void Main(){SimpleDelegate del = PrintMessage;IAsyncResult asyncResult = del.BeginInvoke("Hello, World!", null, null); // 异步调用Console.WriteLine("Main method continues...");// 等待异步调用完成并获取结果string result = del.EndInvoke(asyncResult);Console.WriteLine(result);}
}

输出结果

Main method continues...
Message: Hello, World!
Done
using System;
using System.Threading;public class Program
{public static void Main(){// 定义委托Func<int> func = () =>{Thread.Sleep(2000);Console.WriteLine("Asynchronous operation completed.");return 42;};// 异步调用IAsyncResult result = func.BeginInvoke(null, null);// 执行其他操作Console.WriteLine("Performing other tasks while waiting...");// 等待异步操作完成并获取结果int returnValue = func.EndInvoke(result);Console.WriteLine($"Return value: {returnValue}");}
}

三、DynamicInvoke 方法

1. 定义

DynamicInvoke 是一种动态调用委托的方法,允许以任意类型的参数调用委托,而不需要指定具体的参数类型。
Invoke 不同,DynamicInvoke 可以在运行时动态地确定要调用的方法,并且可以处理参数和返回值的类型。

2. 特点

  • 灵活性高:可以在运行时动态确定参数类型和数量。
  • 性能较低:由于需要进行类型检查和转换,性能通常低于直接调用(如 InvokeBeginInvoke)。

3. 示例代码

using System;public class Program
{public delegate string SimpleDelegate(string message);public static string PrintMessage(string message){Console.WriteLine($"Message: {message}");return "Done";}public static void Main(){SimpleDelegate del = PrintMessage;object[] args = new object[] { "Hello, DynamicInvoke!" };string result = (string)del.DynamicInvoke(args); // 动态调用Console.WriteLine(result);}
}

输出结果

Message: Hello, DynamicInvoke!
Done
using System;public class Program
{public static void Main(){// 定义委托Func<int, int, int> add = (a, b) => a + b;// 动态调用object result = add.DynamicInvoke(2, 3);Console.WriteLine($"Result: {result}");	//输出:Result: 5}
}

四、比较与总结

1. 概览

方法调用方式是否阻塞当前线程参数类型要求性能主要应用场景
Invoke同步固定简单同步调用
BeginInvoke异步固定中等异步调用
EndInvoke同步固定中等获取异步调用结果
DynamicInvoke动态动态运行时动态调用

2. 具体区别与联系

  1. Invoke vs BeginInvoke

    • Invoke 是同步调用,会阻塞当前线程直到方法执行完成;BeginInvoke 是异步调用,不会阻塞当前线程。
    • Invoke 适用于需要立即得到结果的场景;BeginInvoke 适用于需要提高响应速度、避免阻塞主线程的场景。
  2. BeginInvoke vs EndInvoke

    • BeginInvoke 用于启动异步调用,返回一个 IAsyncResult 对象;EndInvoke 用于获取异步调用的结果或等待异步调用完成。
    • 在使用 BeginInvoke 后,必须调用 EndInvoke 来确保资源释放和获取结果。
  3. DynamicInvoke

    • DynamicInvoke 提供了极大的灵活性,但代价是性能较低,因为它需要在运行时进行类型检查和转换。
    • 适用于需要在运行时动态确定参数类型和数量的场景。

3. 实际应用建议

  • 同步调用:如果你需要立即得到结果并且不关心阻塞当前线程,使用 Invoke
  • 异步调用:如果你希望在不阻塞当前线程的情况下执行某个操作,使用 BeginInvokeEndInvoke
  • 动态调用:如果你需要在运行时动态确定参数类型和数量,使用 DynamicInvoke,但要注意其性能开销。
  • 实时数据展示:用 Control.BeginInvoke 异步更新 UI。
  • 批量计算任务:通过 BeginInvoke 分发任务到线程池。
  • 插件系统:结合 DynamicInvoke 实现动态方法调用。

通过合理选择调用方式,可以在保证线程安全的同时提升程序性能与响应速度。

4. 联系与协作

  • 异步调用链
    BeginInvoke 启动异步任务 → 通过 IAsyncResult 监控状态 → EndInvoke 获取结果,形成完整的异步流程。
  • 线程安全
    UI 控件操作中,Control.InvokeControl.BeginInvoke 强制在 UI 线程执行委托,避免跨线程访问异常。
  • 替代方案
    现代 C# 推荐使用 Taskasync/await 替代 BeginInvoke/EndInvoke,因其代码可读性更高且资源管理更安全。

5. 注意事项

  1. 性能开销
    BeginInvoke 依赖线程池,频繁调用可能导致资源竞争;DynamicInvoke 的反射机制效率较低,慎用于高频场景。
  2. 异常处理
    • InvokeDynamicInvoke 的异常直接传播,需用 try-catch 包裹;
    • BeginInvoke 的异常需在 EndInvoke 中捕获。
  3. 代码优化
    发布模式下,编译器可能内联方法,可通过 [MethodImpl(MethodImplOptions.NoInlining)] 保留堆栈信息。

结语

回到目录页:C#/.NET 知识汇总
希望以上内容可以帮助到大家,如文中有不对之处,还请批评指正。


http://www.ppmy.cn/devtools/166678.html

相关文章

【从零开始学习计算机科学】操作系统(二)进程与线程

【从零开始学习计算机科学】操作系统(二)进程与线程 进程PCB进程状态执行模式进程的创建进程的切换进程的撤销进程的阻塞和唤醒进程管理1,写时复制(写时拷贝)2,线程池3,套接字(Socket)4,远程过程调用5,远程方法调用线程进程 进程这个概念体现在并发执行的程序模型中…

Go语言Viper配置详解:conf库优雅解析实战

在现代软件开发中&#xff0c;配置文件是不可或缺的一部分。无论是 YAML、JSON 还是 TOML&#xff0c;如何高效地将这些格式解析到 Go 结构体中&#xff0c;同时支持动态更新&#xff0c;一直是开发者的痛点。好消息是&#xff0c;基于 Viper 封装的 conf 库提供了一个简洁而强…

正则表达式(2)匹配规则

正则表达式的匹配规则定义了如何识别字符串中的特定模式。这些规则包括字符类匹配、元字符匹配、数量词、字符转义和分组。 字符类匹配 字符类匹配允许你指定一个字符集合&#xff0c;并匹配该集合中的任意单个字符。这是通过方括号 [] 来实现的。 简单字符类&#xff1a;[abc…

如何在 React 中实现错误边界?

在 React 中实现错误边界 错误边界是 React 提供的一种机制&#xff0c;用于捕获子组件树中的 JavaScript 错误&#xff0c;并展示回退 UI。它可以帮助开发者更好地处理错误&#xff0c;提升用户体验。本文将详细介绍如何在 React 中实现错误边界&#xff0c;包括其工作原理、…

git subtree更新子仓库的方式

在使用 git subtree 管理的仓库中&#xff0c;子仓库的更新可以通过以下步骤完成&#xff1a; 1. 更新子仓库到主仓库 如果子仓库有新的更新&#xff0c;可以将这些更新拉取到主仓库中&#xff1a; git subtree pull --prefix<子仓库路径> <子仓库地址> <分支…

c#面试题整理4

1.stirng str"",string strnull&#xff0c;俩者有何区别 空字符串占有存储控件&#xff0c;null不占用 2.class与struct的异同 异同class 可继承 引用类型 1.都可以定义方法字段 2.都可实例化&#xff0c;与类的使用几乎一样 struct 不可继承 值类型 只能声明带…

有必要使用 Oracle 向量数据库吗?

向量数据库最主要的特点是让传统的只能基于具体值/关键字的数据检索&#xff0c;进化到了可以直接基于语义的数据检索。这在AI时代至关重要&#xff01; 回到标题问题&#xff1a;是否有必要使用 Oracle 向量数据库&#xff1f; 这实际还要取决于你的具体应用需求。 客观来讲…

Python的那些事第四十三篇:功能强大的测试框架pytest

pytest:功能强大的测试框架 摘要 本文旨在深入探讨 pytest 这一功能强大的测试框架。pytest 具有简单易用、功能丰富等特点,支持分布式测试、自动化测试用例发现等功能。本文将从 pytest 的基本概念、主要功能、使用方法等多个方面进行详细阐述,并通过具体的代码示例和表格…