第二十章:异步和文件I/O.(十五)

在异步操作中,直接访问Task对象的Result属性看似简便,实则可能导致UI线程阻塞,造成用户体验下降及潜在的死锁问题。正确做法是使用ContinueWith或await运算符确保异步操作在不影响主线程的情况下完成。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

不要阻止UI线程!
有时,有一种诱惑可以避免ContinueWith的麻烦,甚至可以通过阻止用户界面线程直到后台进程完成来等待更轻微的麻烦。 也许您知道后台进程会很快完成,并且用户无论如何都不会做任何事情直到完成。 有什么危害?
不要这样做! 它不仅对用户不礼貌,而且可能会在您的应用程序中引入微妙的错误。
让我们举个例子:在TextFileAsyncPage的代码隐藏文件中,OnFileListViewItemSelected处理程序具有以下代码来读取文件并在编辑器中设置内容:

fileEditor.Text = await fileHelper.ReadTextAsync((string)args.SelectedItem);

您可能已经意外地或者通过实验发现,在这样的语句中,您可以省略await运算符并只访问从ReadTextAsync返回的Task 对象的Result属性。 Result属性是要读取的文件的内容:

fileEditor.Text = fileHelper.ReadTextAsync((string)args.SelectedItem).Result;

代码似乎很好,它甚至可能工作。 但它的工作方式并不好。 此语句将阻止用户界面线程,直到ReadTextAsync方法完成并且Result可用。 在此期间,用户界面将无响应。
此外,如果在FileHelper中的ReadTextAsync实现中未使用ConfigureAwait(false),则该ReadTextAsync方法将需要切换到用户界面线程,以便在每个await运算符之后恢复执行。 但是当它尝试切换回用户界面线程时,UI线程将无法使用,因为它在TextFileAsyncPage中的ReadTextAsync调用中被阻止,并且会产生典型的死锁。 该程序将完全停止执行。
规则很简单:对每个异步方法使用ContinueWith或await。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值