当前位置: 首页 > 工具软件 > dispatcher > 使用案例 >

WPF Dispatcher.BeginInvoke 和 Dispatcher.InvokeAsync 之间的区别

子车宏浚
2023-12-01

介绍

在.NET 4.5中,向 Dispatcher 类添加了InvokeAsync 方法。该方法与传统的BeginInvoke 方法类似,但异常处理有很大不同。因此,这不仅仅是使用新的 InvokeAsync 方法而不是旧的 BeginInvoke 方法的问题,您必须根据处理异常的方式来不同地使用它。

开始调用方法

在 BeginInvoke 方法中,如果作为参数传递的委托引发异常,则会将异常通知给Dispatcher.UnhandledExceptionApplication.DispatcherUnhandledException 事件的处理程序。此外,如果这些事件不处理异常,AppDomain.UnhandledException 事件将通知您该异常。

BeginInvoke 方法不通过返回值通知异常。因此,即使像下面的代码那样编写 await,也无法捕获异常。


async Task Example(Dispatcher diapatcher)
{
    try
    {
        await diapatcher.BeginInvoke((Action)(() =>
        {
            
            throw new InvalidOperationException("error");
        }));
    }
    catch (Exception ex)
    {
       
    }
}

调用异步方法

如果作为参数传递的委托引发异常,则 InvokeAsync 方法不会发出诸如 Dispatcher.UnhandledException 事件之类的事件。相反,您将通过返回值收到异常通知。

// InvokeAsync
async Task Example(Dispatcher diapatcher)
{
    try
    {
        await diapatcher.InvokeAsync(() =>
        {
            throw new InvalidOperationException("error");
        });
    }
    catch (Exception ex)
    {
       
    }
}

如果您忘记像下面的代码那样编写 await,则不会将异常通知给调用者。因此,即使写了try/catch,也无法捕捉到异常。


void Example(Dispatcher diapatcher)
{
    try
    {
        
        diapatcher.InvokeAsync(() =>
        {
            throw new InvalidOperationException("error");
        });
    }
    catch (Exception ex)
    {
       
    }
}

概括

在 BeginInvoke 方法中,作为参数传递的委托引发的异常将作为 Dispatcher.UnhandledException 事件或 AppDomain.UnhandledException 事件进行通知。另一方面,在 InvokeAsync 方法中,抛出的异常通过返回值通知调用者。

因此,对于难以向调用者返回异常的异步方法(返回为 void 的异步方法),请使用 BeginInvoke 方法。另一方面,在通知调用者异常的异步方法(返回非 void 的异步方法)的情况下,使用 InvokeAsync 方法。

 类似资料: