扩展方法不等同于直接调用
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 同步上下文被捕获,并且您遇到了死锁。
我会说我拥有的以下两个代码片段是等效的,但它们不是。
以下工作正常:
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 同步上下文被捕获,并且您遇到了死锁。