在上周解决“博客程序异步化改造之后遭遇的性能问题”的过程中,我们干了一件自以为很有成就感的事——在表现层(MVC与WebForms)将所有使用await的地方都加上了ConfigureAwait(false),比如下面代码:

var taskCategories = GetCategoriesAsync();
model.Posts = await GetPostsAsync(model).ConfigureAwait(false);
model.Paging.TotalCount = await taskTotalCount.ConfigureAwait(false);
model.HeadlineHtml = await taskHeadlineHtml.ConfigureAwait(false);
model.Categories = await taskCategories.ConfigureAwait(false);

干完之后才恍然大悟,我们“出色”地完成了一件傻事,性能问题并没有得到解决,最终发现问题的真正原因是我们修改的EnyimMemcaced代码存在内存泄漏问题

犯傻之后,郁闷之时,意外地发现竟然有一些收获可以分享,于是自伤的心有了些许安慰。

在干这次傻事之前,我们分不清默认的ConfigureAwait(continueOnCapturedContext:true)与ConfigureAwait(false)的区别,只知道一个是在异步执行时捕获上下文,一个是在异步执行时不捕获上下文。更不知道ConfigureAwait(false)会带来什么影响?

傻过之后,我们对此多了一点了解:

1)当ConfigureAwait(true),代码由同步执行进入异步执行时,当前同步执行的线程上下文信息(比如HttpConext.Current,Thread.CurrentThread.CurrentCulture)就会被捕获并保存至SynchronizationContext中,供异步执行中使用,并且供异步执行完成之后(await之后的代码)的同步执行中使用(虽然await之后是同步执行的,但是发生了线程切换,会在另外一个线程中执行「ASP.NET场景」)。这个捕获当然是有代价的,当时我们误以为性能问题是这个地方的开销引起,但实际上这个开销很小,在我们的应用场景不至于会带来性能问题。

2)当Configurewait(flase),则不进行线程上下文信息的捕获,async方法中与await之后的代码执行时就无法获取await之前的线程的上下文信息,在ASP.NET中最直接的影响就是HttpConext.Current的值为null。

我们在犯傻过程中,工作量最大的就是处理HttpConext.Current为null的情况。

由于之前写代码时的幼稚与偷懒,造成了很多地方用HttpConext.Current去获取http请求相关信息。在异步化改造之后,HttpConext.Current也遍布在很多aync方法中。Configurewait(flase)之后,NullReferenceException如雨后春笋。

针对这样的窘境,我们只能一个个修改代码,通过方法参数传递所需要的HttpConext信息,取代原先的HttpConext.Current“绿色通道”访问方式。

还有些地方根本不需要HttpConext.Current,只是因为当初的幼稚,比如Server.MapPath(),Server.UrlEncode(),进行了这样的更改:

System.Web.Hosting.HostingEnvironment.MapPath();
HttpUtility.UrlEncode();

在处理HttpConext.Current为null的情况中,我们遇到了一个棘手的问题,它出现在MVC与WebForms混用的场景——在MVC Controller中加载WebForms中的UserControl(也就是让UserControl直接Render为字符串)。

之前我们在MVC中是这样处理的:

page.Controls.Add(commentControl);
using (var sw = new StringWriter())
{
HttpContext.Current.Server.Execute(page, sw, true);
commentsHtml = sw.ToString();
}

可是现在HttpContext.Current为null,不得不改成这样:

page.Controls.Add(commentControl);
var sb = new StringBuilder();
using (var sw = new StringWriter(sb))
{
using (var htw = new HtmlTextWriter(sw))
{
commentControl.RenderControl(htw);
commentsHtml = sb.ToString();
}
}

改好之后发现,只要.ascx中用到了HyperLink控件并访问NavigateUrl属性,就会出现NullReferenceException。原来HyperLink.NavigateUrl中调用了ResolveClientUrl方法,而ResolveClientUrl时会访问Context.Request.ClientBaseDir.VirtualPathString,而Context为null。

在这个地方折腾了不少时间,后来绕道解决了这个问题,用Attributes.Add取代HyperLink.NavigateUrl,比如:

hl.Attributes.Add("href", "http://www.cnblogs.com/");

在WebForms的UserControl中添加ConfigureAwait(false)时,我们开始时以为await之后的代码如果访问Context,也会引发NullReferenceException,而事实表明不会引发。因为在ASP.NET实例化UserControl时会将HttpContext的值传给UserControl的Context属性,所以在UserControl无需通过线程上文获取HttpContext信息(Page的情况也一样)。

另外一个受影响的地方就是线程的CurrentCulture设置,之前我们是在Application_BeginRequest中处理的,代码如下:

protected void Application_BeginRequest(Object sender, EventArgs e)
{
CultureInfo newci = new CultureInfo("zh-CN");
newci.DateTimeFormat.DayNames = new string[] { "日", "一", "二", "三", "四", "五", "六" };
newci.DateTimeFormat.FirstDayOfWeek = DayOfWeek.Sunday;
System.Threading.Thread.CurrentThread.CurrentCulture = newci;
}

但是由于ConfigureAwait(false),异步执行中的线程切换会造成CurrentCulture丢失。

解决方法很简单,直接设置所有线程的CurrentCulture,代码如下:

protected void Application_Start(Object sender, EventArgs e)
{
var newCulture = new CultureInfo("zh-CN");
newCulture.DateTimeFormat.DayNames = new string[] { "日", "一", "二", "三", "四", "五", "六" };
newCulture.DateTimeFormat.FirstDayOfWeek = DayOfWeek.Sunday;
CultureInfo.DefaultThreadCurrentCulture = newCulture;
}

ConfigureAwait(false)的使用经验值得分享的就这些。

在这次异步化改造过程中,不仅加ConfigureAwait(false)是干傻事,整个异步化改造就是一件大傻事。改造过程很艰辛,多次犹豫是不是要这么彻底地异步化,最终还是坚持了下来。当解决了内存泄漏问题之后,如释重负!异步化改造效果明显,超出了预期——响应性能与吞吐能力都得到了提升。过程中所有的煎熬与痛苦都被成功后涌上心头的那种兴奋所秒杀。

【参考资料】

Best practice to call ConfigureAwait for all server-side code

It's All About the SynchronizationContext

走进异步世界-犯傻也值得分享:ConfigureAwait(false)使用经验分享的更多相关文章

  1. 走进异步世界:EnyimMemcached异步化改造引起的内存泄漏

    6月30日我们发布了异步化改造后的博客程序之后,出现了高内存.高CPU.高线程数的不理想情况. 经过一周的追查,终于水落日出——引起不理想情况的根源是我们修改过的EnyimMemcached代码存在内 ...

  2. [C#] 走进异步编程的世界 - 开始接触 async/await

    走进异步编程的世界 - 开始接触 async/await 序 这是学习异步编程的入门篇. 涉及 C# 5.0 引入的 async/await,但在控制台输出示例时经常会采用 C# 6.0 的 $&qu ...

  3. [C#] 走进异步编程的世界 - 剖析异步方法(上)

    走进异步编程的世界 - 剖析异步方法(上) 序 这是上篇<走进异步编程的世界 - 开始接触 async/await 异步编程>(入门)的第二章内容,主要是与大家共同深入探讨下异步方法. 本 ...

  4. [C#] 走进异步编程的世界 - 剖析异步方法(下)

    走进异步编程的世界 - 剖析异步方法(下) 序 感谢大家的支持,这是昨天发布<走进异步编程的世界 - 剖析异步方法(上)>的补充篇. 目录 异常处理 在调用方法中同步等待任务 在异步方法中 ...

  5. [C#] 走进异步编程的世界 - 在 GUI 中执行异步操作

    走进异步编程的世界 - 在 GUI 中执行异步操作 [博主]反骨仔 [原文地址]http://www.cnblogs.com/liqingwen/p/5877042.html 序 这是继<开始接 ...

  6. 走进异步编程的世界 - 开始接触 async/await

    [C#] 走进异步编程的世界 - 开始接触 async/await   走进异步编程的世界 - 开始接触 async/await 序 这是学习异步编程的入门篇. 涉及 C# 5.0 引入的 async ...

  7. [C#] 走进异步编程的世界 - 开始接触 async/await(转)

    原文链接:http://www.cnblogs.com/liqingwen/p/5831951.html 走进异步编程的世界 - 开始接触 async/await 序 这是学习异步编程的入门篇. 涉及 ...

  8. 走进异步编程的世界 - 开始接触 async/await(转)

    序 这是学习异步编程的入门篇. 涉及 C# 5.0 引入的 async/await,但在控制台输出示例时经常会采用 C# 6.0 的 $"" 来拼接字符串,相当于string.Fo ...

  9. 走进异步编程的世界--async/await项目使用实战

    起因:今天要做一个定时器任务:五分钟查询一次数据库发现超时未支付的订单数据将其状态改为已经关闭(数据量大约100条的情况) 开始未使用异步: public void SelfCloseGpPayOrd ...

随机推荐

  1. Spring-Context之七:使用p-namesapce和c-namespace简化bean的定义

    在Spring中定义bean的方式多种多样,即使使用xml的方式来配置也能派生出很多不同的方式. 比如如下的bean定义: 1 2 3 4 5 6 7 8 9 10 11 12 <beans x ...

  2. C++虚函数表

    大家知道虚函数是通过一张虚函数表来实现的.在这个表中,主要是一个类的虚函数的地址表,这张表解决了继承.覆盖的问题,其内容真是反应实际的函数.这样,在有虚函数的类的实例中,这个表分配在了这个实例的内存中 ...

  3. Bootstrap~多级导航(级联导航)的实现

    回到目录 在bootstrap官方来说,导航最多就是两级,两级以上是无法实现的,大叔找了一些第三方的资料,终于找到一个不错的插件,使用上和效果上都还不错,现在和大家分享一下 插件地址:http://v ...

  4. Session自定义存储及分布式存储

    默认情况下,PHP 使用内置的文件会话保存管理器(files)来完成会话的保存.我们无需设置,PHP默认将session以文件的形式保存到服务器. 通过调用函数 session_start() 即可手 ...

  5. Java的概述以及语法

    Java的语法分为标示符和数据类型 Java的概述: 一些手打的: long l = 12345; //隐式转换 int a = (int)121234567L; //强制转换 float f =12 ...

  6. 让DB2跑得更快——DB2内部解析与性能优化

    让DB2跑得更快——DB2内部解析与性能优化 (DB2数据库领域的精彩强音,DB2技巧精髓的热心分享,资深数据库专家牛新庄.干毅民.成孜论.唐志刚联袂推荐!)  洪烨著 2013年10月出版 定价:7 ...

  7. Chrome同步最新host文件IP列表

    使用Chrome的童靴是不是很多都碰到同步问题呢?网上查来查去的都是给些host文件的修改,可是都是几年前的东西,地址都不对了,想想还是自己找到需要解析的域名的IP地址吧 步骤: 1.DNS设置为8. ...

  8. JAVA-集合作业-已知有十六支男子足球队参加2008 北京奥运会。写一个程序,把这16 支球队随机分为4 个组。采用List集合和随机数

    第二题 已知有十六支男子足球队参加2008 北京奥运会.写一个程序,把这16 支球队随机分为4 个组.采用List集合和随机数 2008 北京奥运会男足参赛国家: 科特迪瓦,阿根廷,澳大利亚,塞尔维亚 ...

  9. 那些年我们写过的T-SQL(下篇)

    下篇的内容很多都会在工作中用到,尤其是可编程对象,那些年我们写过的存储过程,有木有?到目前为止很多大型传统企业仍然很依赖存储过程.这部分主要难理解的部分是事务和锁机制这块,本文会进行简单的阐述.虽然很 ...

  10. lufylegend游戏引擎

    lufylegend游戏引擎介绍:click 这个链接我觉得已经很详细的介绍了这个引擎. 所以以下我只说说一些简单的游戏代码过程. 首先从canvas做游戏叙述起: 这是一个让人很熟悉的简单小游戏,网 ...