不要在 ASP.NET 4.5 Beta 的 Page 类事件上直接使用 async 与 await
欢迎到我的博客中阅读独立版本:http://www.dozer.cc/2012/03/async-and-await-in-asp-net-beta/
发现问题
在我的上一篇文章《async 与 await 在 Web 下的应用》中,我提到了 asp.net 4.5 在 Web.Config 中的一个奇怪配置:
1 2 3 | < appSettings > < add key="aspnet:UseTaskFriendlySynchronizationContext" value="true" /> </ appSettings > |
在 Stack Overflow 上提问后,终于有人回答我了。
看了别人的回复后,才发现了我上篇文章中的问题。
下面代码中的这种用法是错误的:
1 2 3 4 5 6 7 8 | protected async void Page_Load( object sender, EventArgs e) { WebClient client = new WebClient(); var result1 = await client.DownloadStringTaskAsync( "http://www.website.com" ); WebClient client2 = new WebClient(); var result2 = await client.DownloadStringTaskAsync(result1); //do more } |
在事件上直接使用 async 引发的错误
代码段一:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 | public partial class WebForm1 : System.Web.UI.Page { protected string Msg { get ; set ; } protected async void Page_Load( object sender, EventArgs e) { using (WebService service = new WebService()) { Msg = await service.Method1TaskSync(); } } protected async void Button_Test_Click( object sender, EventArgs e) { using (WebService service = new WebService()) { Msg = await service.Method2TaskSync(); } } } |
试问,最后的 Msg 的值是什么?应该是哪个方法的返回值?
如果去掉异步,那答案肯定是 Method2。那加上异步后呢?
这里用的是 async 和 await 来实现了异步,所以逻辑上的先后次序应该和代码上的先后次序一样。
但是上述代码两个事件会一起执行!导致了一定的问题!
总结一下上面代码的问题:当页面中的 Page_Load 事件和别的事件都用了 async 和 await 后会出现执行次序错误、死锁等问题。它们并不会按次序执行。
代码段二:
1 2 3 | < appSettings > < add key="aspnet:UseTaskFriendlySynchronizationContext" value="true" /> </ appSettings > |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 | <%@ Page Language="C#" AutoEventWireup="true" CodeBehind="WebForm1.aspx.cs" Inherits="AsyncAwait.WebForm1" Async="true" %> <! DOCTYPE html> < html xmlns="http://www.w3.org/1999/xhtml"> < head runat="server"> < title ></ title > </ head > < body > < form id="form1" runat="server"> < div > <%:Msg %> </ div > </ form > </ body > </ html > |
后端代码和上面一样的代码,只不过把 UseTaskFriendlySynchronizationContext 的配置改成了 true,并且把数据显示到了页面上。
执行后发现:根本无法显示内容,页面在异步执行结束前就已经输出完毕了。
UseTaskFriendlySynchronizationContext 的作用和错误引发的原因
其实在老外的回答中已经说明了全部,我这里主要是翻译+精简一下。
UseTaskFriendlySynchronizationContext 的作用:
之前版本的 asp.net 所使用的异步不符合 CLR 的规范,而只有 RegisterAsyncTask 这个方法是符合 CLR 规范的。
所以 asp.net 4.5 中,加入这个新的配置是为了禁用掉之前不符合约定的功能,只要把这个配置设置为了 true,别的异步方案全部会失效。(代码段二主要就是演示了这个现象)
引发错误的原因:
async 和 await 关键字在底层主要是利用 SynchronizationContext 来实现了异步。(具体原理我也没研究过)
而这个方案首先不符合 CLR 规范,另外也会引起很多问题。(代码段一主要就是演示了其中一个问题)
目前正确的写法
首先,建议把 UseTaskFriendlySynchronizationContext 设置为 true。
另外,正确的写法如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 | public partial class WebForm1 : System.Web.UI.Page { protected string Msg { get ; set ; } protected void Page_Load( object sender, EventArgs e) { RegisterAsyncTask( new PageAsyncTask(Method1)); } private async Task Method1() { using (WebService service = new WebService()) { Msg = await service.HelloWorldTaskSync(); } } protected void Button_Test_Click( object sender, EventArgs e) { RegisterAsyncTask( new PageAsyncTask(Method2)); } private async Task Method2() { using (WebService service = new WebService()) { Msg = await service.HelloWorldTaskSync(); } } } |
如果需要写异步,一定要用 RegisterAsyncTask 方法,实测证明,支持多次调用,而且会按次序执行。
老外说了,他们也想直接在事件上加 async 来写,但是由于技术原因并没有实现,希望在正式版或者未来的版本中可以实现吧!
参考资料:
http://social.msdn.microsoft.com/Forums/en-NZ/async/thread/b2e8c51e-2808-46d0-92e9-b825321d0af8
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
· 探究高空视频全景AR技术的实现原理
· 理解Rust引用及其生命周期标识(上)
· 物流快递公司核心技术能力-地址解析分单基础技术分享
· .NET 10首个预览版发布:重大改进与新特性概览!
· AI与.NET技术实操系列(二):开始使用ML.NET
· 单线程的Redis速度为什么快?
· 展开说说关于C#中ORM框架的用法!