- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
我正在更新一个库,该库具有在 .NET 3.5 中构建的 API 图面。因此,所有方法都是同步的。我无法更改 API(即,将返回值转换为任务),因为这需要所有调用方都进行更改。所以我只剩下如何以同步方式最好地调用异步方法。这是在 ASP.NET 4、ASP.NET Core 和 .NET/.NET Core 控制台应用程序的上下文中。
我可能还不够清楚 - 情况是我有不支持异步的现有代码,我想使用新的库,例如 System.Net.Http 和仅支持异步方法的 AWS SDK。所以我需要弥合差距,并能够拥有可以同步调用但随后可以在其他地方调用异步方法的代码。
我读了很多书,这个问题已经被问过很多次了。
Calling async method from non async method
Synchronously waiting for an async operation, and why does Wait() freeze the program here
Calling an async method from a synchronous method
How would I run an async Task<T> method synchronously?
Calling async method synchronously
How to call asynchronous method from synchronous method in C#?
问题是大多数答案都不一样!我见过的最常见的方法是使用 .Result,但这可能会导致死锁。我已经尝试了以下所有方法,它们都有效,但我不确定哪种方法是避免死锁的最佳方法,具有良好的性能,并且与运行时配合得很好(在尊重任务调度程序、任务创建选项等方面) ).有确定的答案吗?什么是最好的方法?
private static T taskSyncRunner<T>(Func<Task<T>> task)
{
T result;
// approach 1
result = Task.Run(async () => await task()).ConfigureAwait(false).GetAwaiter().GetResult();
// approach 2
result = Task.Run(task).ConfigureAwait(false).GetAwaiter().GetResult();
// approach 3
result = task().ConfigureAwait(false).GetAwaiter().GetResult();
// approach 4
result = Task.Run(task).Result;
// approach 5
result = Task.Run(task).GetAwaiter().GetResult();
// approach 6
var t = task();
t.RunSynchronously();
result = t.Result;
// approach 7
var t1 = task();
Task.WaitAll(t1);
result = t1.Result;
// approach 8?
return result;
}
最佳答案
So I'm left with how to best call async methods in a synchronous way.
首先,这是一件可以做的事情。我之所以这么说,是因为在 Stack Overflow 上,不考虑具体案例,将其作为一种笼统的声明指出这是魔鬼的行为是很常见的。
正确性不需要一直异步。阻塞一些异步的东西以使其同步会产生性能成本,这可能很重要,也可能完全无关紧要。视具体情况而定。
死锁来自两个线程试图同时进入同一个单线程同步上下文。任何避免这种情况的技术都能可靠地避免阻塞引起的死锁。
在您的代码片段中,所有对 .ConfigureAwait(false)
的调用都是毫无意义的,因为没有等待返回值。 ConfigureAwait
返回一个结构,该结构在等待时表现出您请求的行为。如果简单地删除该结构,它什么都不做。
RunSynchronously
使用无效,因为并非所有任务都能以这种方式处理。此方法适用于基于 CPU 的任务,在某些情况下可能无法工作。
.GetAwaiter().GetResult()
与 Result/Wait()
的不同之处在于它模仿了 await
异常传播行为。你需要决定你是否想要那个。 (因此研究该行为是什么;无需在此重复。)如果您的任务包含单个异常,那么 await
错误行为通常很方便并且几乎没有缺点。如果有多个异常,例如来自失败的 Parallel
循环,其中多个任务失败,则 await
将丢弃除第一个异常之外的所有异常。这使调试变得更加困难。
所有这些方法都具有相似的性能。他们将以一种或另一种方式分配操作系统事件并阻止它。那是昂贵的部分。与此相比,其他机器相当便宜。我不知道哪种方法绝对最便宜。
如果抛出异常,那将是最昂贵的部分。在 .NET 5 上,在快速 CPU 上以每秒最多 200,000 个的速度处理异常。深度堆栈速度较慢,并且任务机制倾向于重新抛出异常,使其成本成倍增加。有一些方法可以在不重新抛出异常的情况下阻止任务,例如 task.ContinueWith(_ => { }, TaskContinuationOptions.ExecuteSynchronously).Wait();
。
我个人喜欢 Task.Run(() => DoSomethingAsync()).Wait();
模式,因为它明确地避免了死锁,它很简单并且没有隐藏一些异常 GetResult()
可能会隐藏。但是您也可以使用 GetResult()
。
关于c# - 从非异步代码调用异步方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40324300/
我尝试理解[c代码 -> 汇编]代码 void node::Check( data & _data1, vector& _data2) { -> push ebp -> mov ebp,esp ->
我需要在当前表单(代码)的上下文中运行文本文件中的代码。其中一项要求是让代码创建新控件并将其添加到当前窗体。 例如,在Form1.cs中: using System.Windows.Forms; ..
我有此 C++ 代码并将其转换为 C# (.net Framework 4) 代码。有没有人给我一些关于 malloc、free 和 sprintf 方法的提示? int monate = ee; d
我的网络服务器代码有问题 #include #include #include #include #include #include #include int
给定以下 html 代码,将列表中的第三个元素(即“美丽”一词)以斜体显示的 CSS 代码是什么?当然,我可以给这个元素一个 id 或一个 class,但 html 代码必须保持不变。谢谢
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 7 年前。
我试图制作一个宏来避免重复代码和注释。 我试过这个: #define GrowOnPage(any Page, any Component) Component.Width := Page.Surfa
我正在尝试将我的旧 C++ 代码“翻译”成头条新闻所暗示的 C# 代码。问题是我是 C# 中的新手,并不是所有的东西都像 C++ 中那样。在 C++ 中这些解决方案运行良好,但在 C# 中只是不能。我
在 Windows 10 上工作,R 语言的格式化程序似乎没有在 Visual Studio Code 中完成它的工作。我试过R support for Visual Studio Code和 R-T
我正在处理一些报告(计数),我必须获取不同参数的计数。非常简单但乏味。 一个参数的示例查询: qCountsEmployee = ( "select count(*) from %s wher
最近几天我尝试从 d00m 调试网络错误。我开始用尽想法/线索,我希望其他 SO 用户拥有可能有用的宝贵经验。我希望能够提供所有相关信息,但我个人无法控制服务器环境。 整个事情始于用户注意到我们应用程
我有一个 app.js 文件,其中包含如下 dojo amd 模式代码: require(["dojo/dom", ..], function(dom){ dom.byId('someId').i
我对“-gencode”语句中的“code=sm_X”选项有点困惑。 一个例子:NVCC 编译器选项有什么作用 -gencode arch=compute_13,code=sm_13 嵌入库中? 只有
我为我的表格使用 X-editable 框架。 但是我有一些问题。 $(document).ready(function() { $('.access').editable({
我一直在通过本教程学习 flask/python http://blog.miguelgrinberg.com/post/the-flask-mega-tutorial-part-i-hello-wo
我想将 Vim 和 EMACS 用于 CNC、G 代码和 M 代码。 Vim 或 EMACS 是否有任何语法或模式来处理这种类型的代码? 最佳答案 一些快速搜索使我找到了 this vim 和 thi
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 想改进这个问题?更新问题,使其成为 on-topic对于堆栈溢出。 7年前关闭。 Improve this
这个问题在这里已经有了答案: Enabling markdown highlighting in Vim (5 个回答) 6年前关闭。 当我在 Vim 中编辑包含 Markdown 代码的 READM
我正在 Swift3 iOS 中开发视频应用程序。基本上我必须将视频 Assets 和音频与淡入淡出效果合并为一个并将其保存到 iPhone 画廊。为此,我使用以下方法: private func d
pipeline { agent any stages { stage('Build') { steps { e
我是一名优秀的程序员,十分优秀!