• 不要在 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>
    <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://stackoverflow.com/questions/9562836/whats-the-meaning-of-usetaskfriendlysynchronizationcontext

    http://social.msdn.microsoft.com/Forums/en-NZ/async/thread/b2e8c51e-2808-46d0-92e9-b825321d0af8

    当前标签: Asp.net

     
    Dozer 2012-03-06 19:39 阅读:1205 评论:3
     
    Dozer 2011-01-26 23:47 阅读:1599 评论:7
     
    Dozer 2010-10-19 18:11 阅读:1795 评论:2
     
    Dozer 2010-09-21 02:15 阅读:3886 评论:30
     
    Dozer 2010-02-14 19:03 阅读:319 评论:1
     
  • 相关阅读:
    【转】Tomcat中部署java web应用程序
    【转】如何安装mysql服务
    【转】java_web开发入门
    【转】SVN 查看历史信息
    【转】java编译错误 程序包javax.servlet不存在javax.servlet.*
    【转】MySQL5安装的图解(mysql-5.0.27-win32.zip)
    【转】JAVA变量path , classpth ,java_home设设置作用和作用
    intellij idea 10.5介绍
    Java中的IO与NIO
    javaWeb完成注册功能
  • 原文地址:https://www.cnblogs.com/Leo_wl/p/2399754.html
Copyright © 2020-2023  润新知