gpt4 book ai didi

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

转载 作者:太空狗 更新时间:2023-10-30 00:38:52 24 4
gpt4 key购买 nike

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

以下工作正常:

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 Graph API提供,函数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);
}

最佳答案

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

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) 返回 Task),因此 UI 同步上下文被捕获,并且您遇到了死锁。

关于c# - 扩展方法不等同于直接调用,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37678435/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com