走进异步世界-犯傻也值得分享:ConfigureAwait(false)使用经验分享
发布日期:2021-05-09 01:36:16 浏览次数:24 分类:博客文章

本文共 3730 字,大约阅读时间需要 12 分钟。

在上周解决“”的过程中,我们干了一件自以为很有成就感的事——在表现层(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)是干傻事,整个异步化改造就是一件大傻事。改造过程很艰辛,多次犹豫是不是要这么彻底地异步化,最终还是坚持了下来。当解决了之后,如释重负!异步化改造效果明显,超出了预期——响应性能与吞吐能力都得到了提升。过程中所有的煎熬与痛苦都被成功后涌上心头的那种兴奋所秒杀。

【参考资料】

上一篇:云计算之路-阿里云上:消灭“黑色n秒”第一招——不让CPU空闲
下一篇:2004-2014,博客园十年

发表评论

最新留言

路过,博主的博客真漂亮。。
[***.116.15.85]2025年05月06日 04时23分34秒

关于作者

    喝酒易醉,品茶养心,人生如梦,品茶悟道,何以解忧?唯有杜康!
-- 愿君每日到此一游!

推荐文章

excel中最常用的30个函数_Excel玩转数据分析常用的43个函数! 2023-01-24
flink sql设置并行度_Flink 参数配置和常见参数调优 2023-01-24
go 字符串替换_Go 每日一库之 quicktemplate 2023-01-24
hex editor neo下载_口袋妖怪爆焰黑手机版下载-口袋妖怪爆焰黑手游下载v4.3.0 安卓版... 2023-01-24
hibernate mysql 关联查询_spring-boot hibernate 双向关联查询的坑 2023-01-24
hive 建表_sqoop的使用之导入到hive和mysql 2023-01-24
hp工作站z8装Linux,惠普Z8G4双路最小工作站 2023-01-24
html上传图片直接保存到数据库中,Editor上传图片路径存入数据库中怎么弄? 2023-01-24
html游戏玩不了,WinXP网页游戏玩不了怎么办有哪些解决方法 2023-01-24
html转jsp_JSP详解 2023-01-24
ICLOUD储存空间要升级吗_有人像我一样需要恢复苹果手机icloud空间ios备份时 微信卡住不动了吗(已解决)... 2023-01-24
image unity 原始尺寸_Unity基础教程-对象管理(十一)——生命周期(Growth and Death)... 2023-01-24
iphone打字怎么换行_手持iPhone?你可能并不知道的小技巧! 2023-01-24
jaccard相似度_自然语言处理之文本相似度计算 2023-01-24
java 8 list对象属性判空_java ---- 认识类对象,属性和方法 2023-01-24
java http delete_java积累---HttpDelete请求方式传递参数 2023-01-24
java swing数据库,如何在Java swing中查看数据库结果集 2023-01-24
java xmpp 群聊,使用XMPPFramework openfire创建聊天室 2023-01-24
java 反义_java中一些常用的英语 2023-01-24
java 命令行 class_如何从命令行执行java .class 2023-01-24