- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我读了这篇文章https://blog.stephencleary.com/2012/07/dont-block-on-async-code.html - 但是我看到了一个矛盾:
我知道 UI 线程死锁的问题,因为 UI 线程阻塞等待异步操作完成,但相同的异步操作同步到 UI 线程上下文 - 因此异步操作无法进入 UI 线程,因此 UI 线程不会停止等待。
文章告诉我们解决方法是不要在 UI 线程上阻塞,否则您需要使用 ConfigureAwait(false)
everywhere:
You would have to use for every await in the transitive closure of all methods called by the blocking code, including all third- and second-party code.
然而作者在文章后面写道:
Preventing the Deadlock
There are two best practices (both covered in my intro post) that avoid this situation:
- In your “library” async methods, use
ConfigureAwait(false)
wherever possible.- Don’t block on Tasks; use
async
all the way down.
我在这里看到一个矛盾 - 在“不要这样做”部分中,他写道必须在任何地方使用 ConfigureAwait(false)
将是阻塞 UI 线程的结果 - 但是在他的“最佳实践”列表中,他告诉我们要做到这一点:“尽可能使用 ConfigureAwait(false)
”。 - 尽管我认为“只要有可能”就会排除第三方代码,但在没有第三方代码的情况下,无论是否阻塞 UI 线程,结果都是一样的。
至于我的具体问题,这是我当前在 WPF MVVM 项目中的代码:
private async void ButtonClickEventHandler()
{
WebServiceResponse response = await this.client.PushDinglebopThroughGrumbo();
this.DisplayResponseInUI( response );
}
public class PlumbusWebServiceClient {
private static readonly HttpClient _client = new HttpClient();
public async Task<WebServiceResponse> PushDinglebopThroughGrumbo()
{
try
{
using( HttpResponseMessage response = await _client.GetAsync( ... ) )
{
if( !response.IsSuccessStatusCode ) return WebServiceResponse.FromStatusCode( response.StatusCode );
using( Stream versionsFileStream = await response.Content.ReadAsStreamAsync() )
using( StreamReader rdr = new StreamReader( versionsFileStream ) )
{
return await WebServiceResponse.FromResponse( rdr );
}
}
}
catch( HttpResponseException ex )
{
return WebServiceResponse.FromException( ex );
}
}
}
如果我对文档的理解正确,我应该将 ConfigureAwait(false)
添加到 every await
没有代码的方法中需要在 UI 线程上运行 - 这是我的 PushDinglebopThroughGrumbo
方法中的每个方法,还有 WebServiceResponse.FromResponse
中的所有代码(调用 await StreamReader.ReadLineAsync
)。但是我调用的任何第三方代码也对 StreamReader
执行 await
操作呢?我无法访问他们的源代码,所以那是不可能的。
我也有点推迟必须在任何地方放置 ConfigureAwait(false)
- 我认为 await
关键字的目的是消除显式任务库调用 - 不应该有一个不同的关键字用于 resume-context-free 等待吗? (例如 awaitfree
)。
那么我的代码应该是这样的吗?
(unmodified, same as above)
public class PlumbusWebServiceClient {
private static readonly HttpClient _client = new HttpClient();
public async Task<WebServiceResponse> PushDinglebopThroughGrumbo()
{
try
{
using( HttpResponseMessage response = await _client.GetAsync( ... ).ConfigureAwait(false) ) // <-- here
{
if( !response.IsSuccessStatusCode ) return WebServiceResponse.FromStatusCode( response.StatusCode );
using( Stream versionsFileStream = await response.Content.ReadAsStreamAsync().ConfigureAwait(false) ) // <-- and here
using( StreamReader rdr = new StreamReader( versionsFileStream ) )
{
return await WebServiceResponse.FromResponse( rdr ).ConfigureAwait(false); // <-- and here again, and inside `FromResponse` too
}
}
}
catch( HttpResponseException ex )
{
return WebServiceResponse.FromException( ex );
}
}
}
...我原以为调用 ConfigureAwait(false)
只需要在 PlumbusWebServiceClient
方法中调用最顶层的 await
- 即 GetAsync
调用。
如果我确实需要在任何地方应用它,我可以将它简化为一个扩展方法吗?
public static ConfiguredTaskAwaitable<T> CF<T>(this Task<T> task) {
return task.ConfigureAwait(false);
}
using( HttpResponseMessage response = await _client.GetAsync( ... ).CF() )
{
...
}
...虽然这并不能减轻所有的繁琐。
这是我编写的一些异步代码,可将我的应用程序设置导出到一个简单的文本文件 - 我忍不住认为这感觉不对,这真的是正确的方法吗?
class Settings
{
public async Task Export(String fileName)
{
using( StreamWriter wtr = new StreamWriter( fileName, append: false ) )
{
await ExportSetting( wtr, nameof(this.DefaultStatus ), this.DefaultStatus ).ConfigureAwait(false);
await ExportSetting( wtr, nameof(this.ConnectionString ), this.ConnectionString ).ConfigureAwait(false);
await ExportSetting( wtr, nameof(this.TargetSystem ), this.TargetSystem.ToString("G") ).ConfigureAwait(false);
await ExportSetting( wtr, nameof(this.ThemeBase ), this.ThemeBase ).ConfigureAwait(false);
await ExportSetting( wtr, nameof(this.ThemeAccent ), this.ThemeAccent ).ConfigureAwait(false);
await ExportSetting( wtr, nameof(this.ShowSettingsButton), this.ShowSettingsButton ? "true" : "false" ).ConfigureAwait(false);
await ExportSetting( wtr, nameof(this.ShowActionsColumn ), this.ShowActionsColumn ? "true" : "false" ).ConfigureAwait(false);
await ExportSetting( wtr, nameof(this.LastNameFirst ), this.LastNameFirst ? "true" : "false" ).ConfigureAwait(false);
await ExportSetting( wtr, nameof(this.TitleCaseCustomers), this.TitleCaseCustomers ? "true" : "false" ).ConfigureAwait(false);
await ExportSetting( wtr, nameof(this.TitleCaseVehicles ), this.TitleCaseVehicles ? "true" : "false" ).ConfigureAwait(false);
await ExportSetting( wtr, nameof(this.CheckForUpdates ), this.CheckForUpdates ? "true" : "false" ).ConfigureAwait(false);
}
}
private static async Task ExportSetting(TextWriter wtr, String name, String value)
{
String valueEnc = Uri.EscapeDataString( value ); // to encode line-breaks, etc.
await wtr.WriteAsync( name ).ConfigureAwait(false);
await wtr.WriteAsync( '=' ).ConfigureAwait(false);
await wtr.WriteLineAsync( valueEnc ).ConfigureAwait(false);
}
}
最佳答案
If I understand the document correctly, I should add
ConfigureAwait(false)
to everyawait
that is not in a method that has code that needs to run on the UI thread
是的。 UI 应用程序中的默认行为是 await
之后的代码在 UI 线程上继续。当 UI 线程繁忙,但您的代码不需要访问 UI 时,等待 UI 线程可用是没有意义的。
(注意:这里有意遗漏了一些与此处无关的细节。)
But what about any third-party code I call which also performs
await
operations on theStreamReader
?
只要通过其他方式避免死锁,这只会影响性能,不会影响正确性。第三方代码可能表现不佳的问题并不是一个新问题。
换句话说:遵循两种最佳实践。
I'm also a bit put-off by having to place
ConfigureAwait(false)
everywhere - I thought the point of theawait
keyword was to eliminate explicit task library calls - shouldn't there be a different keyword for resume-context-free awaiting then? (e.g.awaitfree
).
ConfigureAwait
不是 TPL 方法。
await
是通用的,因此它可以用于任意类型,只要它们支持所需的方法。举一个随机的例子,您可以为 Task
添加一个扩展方法来返回一个类型,该类型允许 await
之后的代码在新的专用线程中继续。这不需要带有新关键字的新版本编译器。
但是,是的,这是一个很长的名字。
If I do need to apply it everywhere, could I simplify it to an extension method?
是的,那很好。
Here is some async code I wrote that exports my application's settings to a simple text file - I can't help but think it doesn't feel right, is this really the correct way to do this?
正如我在评论中所写的那样,我自己根本不会使用这种方法……但如果您确实愿意,您可以消除其中的大量代码重复。随着它的消失,它看起来不再那么糟糕了。
/* SettingsCollection omitted, but trivially implementable using
Dictionary<string, string>, NameValueCollection,
List<KeyValuePair<string, string>>, whatever. */
SettingsCollection GetAllSettings()
{
return new SettingsCollection
{
{ nameof(this.DefaultStatus ), this.DefaultStatus },
{ nameof(this.ConnectionString ), this.ConnectionString },
{ nameof(this.TargetSystem ), this.TargetSystem.ToString("G") },
{ nameof(this.ThemeBase ), this.ThemeBase },
{ nameof(this.ThemeAccent ), this.ThemeAccent },
{ nameof(this.ShowSettingsButton), this.ShowSettingsButton ? "true" : "false" },
{ nameof(this.ShowActionsColumn ), this.ShowActionsColumn ? "true" : "false" },
{ nameof(this.LastNameFirst ), this.LastNameFirst ? "true" : "false" },
{ nameof(this.TitleCaseCustomers), this.TitleCaseCustomers ? "true" : "false" },
{ nameof(this.TitleCaseVehicles ), this.TitleCaseVehicles ? "true" : "false" },
{ nameof(this.CheckForUpdates ), this.CheckForUpdates ? "true" : "false" }
};
}
public async Task Export(String fileName)
{
using( StreamWriter wtr = new StreamWriter( fileName, append: false ) )
foreach (var setting in GetAllSettings())
await ExportSetting( wtr, setting.Key, setting.Value ).ConfigureAwait(false);
}
关于c# - 我应该在每个等待的操作上调用 ConfigureAwait(false),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43423709/
#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
我是一名优秀的程序员,十分优秀!