扩展方法不等同于直接调用

Extension method not equivalent to direct call

我会说我拥有的以下两个代码片段是等效的,但它们不是。

以下工作正常:

var entry3 = Task.Run(async () => await entry2.GetMemberGroupsAsync(false)).WaitForResult().FirstOrDefault();

我刚刚将 Task.Run.WaitForResult 链移动到扩展方法中的以下代码不起作用,但会产生死锁:

var entry3 = entry2.GetMemberGroupsAsync(false).RunSynchronouslyAndReturnResult().FirstOrDefault();

public static T RunSynchronouslyAndReturnResult<T>(this Task<T> task)
{
    return Task.Run(async () => await task).WaitForResult();
}

为什么这两个代码片段不等价?

为了完整起见,GetMemberGroupsAsync方法由Microsoft Azure GraphAPI提供,函数WaitForResult定义如下。据我所知,它不会根据来电者姓名或某事做任何不同的事情。像那样:

public static TResult WaitForResult<TResult>(this Task<TResult> task,
                                             bool continueOnCapturedContext = false)
{
    if (task == null)
    {
        throw new ArgumentNullException("task");
    }

    try
    {
        return PreventForDeadLocks(task, continueOnCapturedContext).Result;
    }
    catch (AggregateException ex)
    {
        if (ex.InnerExceptions.Count == 1)
        {
            throw ex.InnerExceptions[0];
        }

        throw;
    }
}

public static async Task<TResult> PreventForDeadLocks<TResult>(this Task<TResult> task,
                                                               bool continueOnCapturedContext = false)
{
    return await task.ConfigureAwait(continueOnCapturedContext: continueOnCapturedContext);
}

在第一种情况下,GetMemberGroupsAsync 是在与 WaitForResult 不同的线程上调用的。

在第二种情况下,它与WaitForResult 在同一个线程上被调用。您只是 在另一个线程上等待

此处的区别在于您的任务在哪个同步上下文中启动。这里:

var entry3 = Task.Run(async () => await entry2.GetMemberGroupsAsync(false)).WaitForResult().FirstOrDefault();

您在 Task.Run 调用中启动异步任务(我的意思是 await entry2.GetMemberGroupsAsync(false)),因此未捕获 UI 同步上下文。但是在这里:

var entry3 = entry2.GetMemberGroupsAsync(false).RunSynchronouslyAndReturnResult().FirstOrDefault();

您在 UI 上下文中隐式启动您的任务(entry2.GetMemberGroupsAsync(false) returns 任务),因此 UI 同步上下文被捕获,并且您遇到了死锁。