- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
根据 this link :
When you are awaiting on a method with await keyword, compiler generates bunch of code in behalf of you. One of the purposes of this action is to handle synchronization with the UI thread. The key
component of this feature is theSynchronizationContext.Current
which gets the synchronization context for the current thread.
SynchronizationContext.Current
is populated depending on the
environment you are in. TheGetAwaiter
method of Task looks up for
SynchronizationContext.Current
. If current synchronization context is not null, the continuation that gets passed to that awaiter will get posted back to that synchronization context.When consuming a method, which uses the new asynchronous language features, in a blocking fashion, you will end up with a deadlock if
you have an availableSynchronizationContext
. When you are consuming such methods in a blocking fashion (waiting on the Task with Wait method or taking the result directly from the Result property of the Task), you will block the main thread at the same time. When eventually the Task completes inside that method in the threadpool, it is going to invoke the continuation to post back to the main thread becauseSynchronizationContext.Current
is available and captured. But there is a problem here: the UI thread is blocked and you have a deadlock!
public class HomeController : Controller
{
public ViewResult CarsSync()
{
SampleAPIClient client = new SampleAPIClient();
var cars = client.GetCarsInAWrongWayAsync().Result;
return View("Index", model: cars);
}
}
public class SampleAPIClient
{
private const string ApiUri = "http://localhost:17257/api/cars";
public async Task<IEnumerable<Car>> GetCarsInAWrongWayAsync()
{
using (var client = new HttpClient())
{
var response = await client.GetAsync(ApiUri);
// Not the best way to handle it but will do the work for demo purposes
response.EnsureSuccessStatusCode();
return await response.Content.ReadAsAsync<IEnumerable<Car>>();
}
}
}
我无法理解上面语句的粗体 部分,但是当我测试上面的代码时,它按预期发生了死锁。但是我还是不明白为什么UI线程会被阻塞?
在这种情况下,可用的 SynchronizationContext
是什么?是 UI 线程吗?
最佳答案
我对此进行了完整解释 in my own blog post , 但在这里重申......
await
默认情况下将捕获当前“上下文”并恢复其 async
该上下文中的方法。此上下文是 SynchronizationContext.Current
除非它是 null
, 在这种情况下它是 TaskScheduler.Current
.
一次一个线程时可能会发生死锁 SynchronizationContext
并且您阻止代表异步代码的任务(例如,使用 Task.Wait
或 Task<T>.Result
)。请注意,导致死锁的是阻塞,而不仅仅是 SynchronizationContext
。 ;适当的解决方案(几乎总是)是使调用代码异步(例如,将 Task.Wait
/Task<T>.Result
替换为 await
)。在 ASP.NET 上尤其如此。
But I still can't understand why the UI thread is blocked?
您的示例在 ASP.NET 上运行;没有 UI 线程。
what is the available SynchronizationContext?
当前SynchronizationContext
应该是 AspNetSynchronizationContext
的实例,表示 ASP.NET 请求的上下文。此上下文一次只允许一个线程。
因此,通过您的示例:
当收到此操作的请求时,CarsSync
将在该请求上下文中开始执行。它继续到这一行:
var cars = client.GetCarsInAWrongWayAsync().Result;
本质上与此相同:
Task<IEnumerable<Car>> carsTask = client.GetCarsInAWrongWayAsync();
var cars = carsTask.Result;
因此,它继续调用 GetCarsInAWrongWayAsync
, 它会一直运行直到它到达第一个 await
(GetAsync
电话)。此时,GetCarsInAWrongWayAsync
捕获其当前上下文(ASP.NET 请求上下文)并返回不完整的 Task<IEnumerable<Car>>
.当 GetAsync
下载完成,GetCarsInAWrongWayAsync
将继续在该 ASP.NET 请求上下文中执行并(最终)完成它已经返回的任务。
但是,一旦GetCarsInAWrongWayAsync
返回未完成的任务,CarsSync
阻塞当前线程,等待该任务完成。请注意,当前线程位于该 ASP.NET 请求上下文中,因此 CarsSync
会阻止 GetCarsInAWrongWayAsync
永远不会恢复执行,导致死锁。
最后一点,GetCarsInAWrongWayAsync
是一个好的方法。如果用ConfigureAwait(false)
就更好了,但这实际上不是错误的。 CarsSync
是导致死锁的方法;它调用Task<T>.Result
是错的。适当的修复是更改 CarsSync
:
public class HomeController : Controller
{
public async Task<ViewResult> CarsSync()
{
SampleAPIClient client = new SampleAPIClient();
var cars = await client.GetCarsInAWrongWayAsync();
return View("Index", model: cars);
}
}
关于c# - 使用 SynchronizationContext 时异步/等待死锁,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34151179/
#include using namespace std; class C{ private: int value; public: C(){ value = 0;
这个问题已经有答案了: What is the difference between char a[] = ?string?; and char *p = ?string?;? (8 个回答) 已关闭
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 7 年前。 此帖子已于 8 个月
除了调试之外,是否有任何针对 c、c++ 或 c# 的测试工具,其工作原理类似于将独立函数复制粘贴到某个文本框,然后在其他文本框中输入参数? 最佳答案 也许您会考虑单元测试。我推荐你谷歌测试和谷歌模拟
我想在第二台显示器中移动一个窗口 (HWND)。问题是我尝试了很多方法,例如将分辨率加倍或输入负值,但它永远无法将窗口放在我的第二台显示器上。 关于如何在 C/C++/c# 中执行此操作的任何线索 最
我正在寻找 C/C++/C## 中不同类型 DES 的现有实现。我的运行平台是Windows XP/Vista/7。 我正在尝试编写一个 C# 程序,它将使用 DES 算法进行加密和解密。我需要一些实
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
有没有办法强制将另一个 窗口置于顶部? 不是应用程序的窗口,而是另一个已经在系统上运行的窗口。 (Windows, C/C++/C#) 最佳答案 SetWindowPos(that_window_ha
假设您可以在 C/C++ 或 Csharp 之间做出选择,并且您打算在 Windows 和 Linux 服务器上运行同一服务器的多个实例,那么构建套接字服务器应用程序的最明智选择是什么? 最佳答案 如
你们能告诉我它们之间的区别吗? 顺便问一下,有什么叫C++库或C库的吗? 最佳答案 C++ 标准库 和 C 标准库 是 C++ 和 C 标准定义的库,提供给 C++ 和 C 程序使用。那是那些词的共同
下面的测试代码,我将输出信息放在注释中。我使用的是 gcc 4.8.5 和 Centos 7.2。 #include #include class C { public:
很难说出这里问的是什么。这个问题是含糊的、模糊的、不完整的、过于宽泛的或修辞性的,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开它,visit the help center 。 已关
我的客户将使用名为 annoucement 的结构/类与客户通信。我想我会用 C++ 编写服务器。会有很多不同的类继承annoucement。我的问题是通过网络将这些类发送给客户端 我想也许我应该使用
我在 C# 中有以下函数: public Matrix ConcatDescriptors(IList> descriptors) { int cols = descriptors[0].Co
我有一个项目要编写一个函数来对某些数据执行某些操作。我可以用 C/C++ 编写代码,但我不想与雇主共享该函数的代码。相反,我只想让他有权在他自己的代码中调用该函数。是否可以?我想到了这两种方法 - 在
我使用的是编写糟糕的第 3 方 (C/C++) Api。我从托管代码(C++/CLI)中使用它。有时会出现“访问冲突错误”。这使整个应用程序崩溃。我知道我无法处理这些错误[如果指针访问非法内存位置等,
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 7 年前。
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,因为
我有一些 C 代码,将使用 P/Invoke 从 C# 调用。我正在尝试为这个 C 函数定义一个 C# 等效项。 SomeData* DoSomething(); struct SomeData {
这个问题已经有答案了: Why are these constructs using pre and post-increment undefined behavior? (14 个回答) 已关闭 6
我是一名优秀的程序员,十分优秀!