ASP.NET MVC 3中的异步 Action 通过两个匹配的方法XxxAsync/XxxCompleted 来定义,这样的异步 Action 只能定义在继承自 AsyncConoller 的类型中。 ASP.NET MVC 4 提供
了一种更加简洁的异步 Action 定义方式,不过为了兼容前一版本,传统的定义方式和AsyncController 类型依然被保留下来。为什么我们需要让 Action 方法异步地执行呢?要回答
这个问题,这就需要了解 ASP.NET 基于线程池 Thread Pool 的请求处理机制。

1. 基于线程池的请求处理机制
ASP.NET 通过线程池的机制处理并发的 HTTP 请求。一个 Web 应用内部维护着一个线程池,当探测到抵达的请求后, ASP.NET 会从池中获取一个空闲的线程来处理它。处理完毕
后,线程不会被回收,而是重新释放到池中。线程池具有一个最大容量,如果创建的线程达到这个上限并且现有的线程均被处于"忙碌"状态,新的 HTTP 请求会被放入一个请求队列
以等待某个线程重新释放到池中。
我们将这些用于处理 HTTP 请求的线程称为工作线程 (Worker Thread) ,而这个线程池自然就叫做工作线程池。

ASP. NET这种基于线程池的请求处理机制主要具有如下两个优势。
  • 工作线程的重用:创建线程的成本虽然不如进程的激活,却也不是一件一蹦而就的事情,
频繁地创建和释放线程会对性能造成极大的损害。线程池机制避免了总是创建新的工作线程来处理每一个请求,被创建的工作线程得到了极大地重用,并最终提高了服务器的吞吐能力。
  • 工作线程数量的限制:资源的有限性决定了服务器处理请求的能力具有一个上限,或者说某台服务器能够处理的请求并发量具有一个临界点,一旦超过这个临界点,整个服务
器将会因不能提供足够的资源而崩溃。由于采用了对工作线程数量具有良好控制的线程池机制, ASP. 并发处理的请求数量不可能超过线程池的最大允许的容量,从而避
免了在高并发情况下工作线程的无限制创建而最终导致整个服务器的崩溃。如果请求处理操作耗时较短,工作线程处理完毕后可以及时地被释放到线程池中以用于
对下一个请求的处理。但是对于比较耗时的操作来说,意味着工作线程将长时间被某个请求独占,如果这样的操作访问比较频繁,在高并发的情况下在线程池中可能找不到空闲的工作
线程用于及时处理最新抵达的请求。如果采用异步的方式来处理这样的耗时请求,工作线程可以让后台线程来接手,而自己可以及时地被释放到线程池中用于进行后续请求的处理,从而提高了整个服务器的吞吐能
力。值得一提的是,异步操作主要用于I1 绑定操作(比如数据库访问和远程服务调用等),而非 CPU 绑定操作,因为异步操作对整体性能的提升来源于当 I/O 设备在处理某个任务的
时候, CPU 可以释放出来去处理另一个任务。如果耗时操作主要依赖于本机 CPU 的运算,用异步方法反而会因为线程调度和线程上下文的切换而影响整体的性能。

2.两种异步 Action 万法的定义

  2.1 XxxAsync/XxxCompleted
  如果使用两个匹配的方法 xxA sync lX xxCompleted 来定义异步 Action ,可以将异步操作实现在 xxA sync 方法中,而将最终内容的响应实现在Xx xCompleted 方法中。 XxxCompleted
可以看成是对 Async 方法的回调,当定义在 xxA sync 方法中的操作以异步方式完成执行后, XxxCompleted 方法会被自动调用。Xx xCompleted 的定义方式和普通的同步 Action
法比较类似。

    public class HomeController : AsyncController
{ //AsyncController
public void ArticleAsync(string name)
{
//ASP.NETMVC 发起异步操作开始 Increment
AsyncManager.OutstandingOperations.Increment();
Task.Factory.StartNew(() =>
{
AsyncManager.Parameters["content"] = "ArticleContent:" + name;
//ASP.NETMVC 发起异步操作结束
AsyncManager.OutstandingOperations.Decrement(); });
Task.Factory.StartNew(() =>
{
AsyncManager.Parameters["content"] = "ArticleContent:" + name;
//ASP.NETMVC 发起异步操作结束 显示操作 AsyndManager.Finish();
AsyncManager.Finish();
}); }
public ActionResult ArticleCompleted(string content)
{
string str = content;
return Content(content);
}
}

  2.2Task 返回值

  如果采用上面的异步 Action 定义方式,意味着我们不得不为一个 Action 定义两个方法,实际上可以通过一个方法来完成对异步 Action 的定义,那就是让 Action 方法返回一个代表
异步操作的 Task 对象。除此之外,以 XxxAsync/XxxCompleted 形式定义的异步 Action 只能出现在继承自 AsyncController 的类型中,而针对 Task 返回值的异步 Action 则对此没有限制。
实际上保留 AsyncController 这个抽象类主要是为了实现对 AS NETMVC 3的向后兼容。

上面通过 XxxAsync /XxxCompleted 形式定义的异步 Action 可以采用如下的定义方式。

  public class HomeController : Controller
{ //Task 返回值
public Task<ActionResult> Article(string name)
{
return Task.Factory.StartNew(() =>
{
AsyncManager.Parameters["content"] = "TaskArticle:" + name;
}).ContinueWith<ActionResult>(task =>
{
string content = (string)AsyncManager.Parameters["content"];
return Content(content);
});
}
}

最新文章

  1. IIS如何设置可以让.aspx后缀的文件直接下载
  2. 如何编写Iveely搜索引擎插件
  3. Java开源数据库管理工具
  4. 手动从浏览器中获取 cookie
  5. VC++入门精通视频教程
  6. scala学习-类与对象
  7. 之前可运行mongodb,后来却不行了显示Unclean shutdown detected mongodb
  8. Vim的设置和使用——编程者
  9. javsscript总结
  10. Docker 安装及问题处理
  11. ZOJ 3702 Gibonacci number
  12. 【下一代核心技术DevOps】:(五)微服务CI与Rancher持续集成
  13. SpringBoot+Jpa+MySql学习
  14. varchar和nvarchar的区别 数据来证明
  15. web应用配置文件详解(web.xml)
  16. 为什么学习Python及Python环境安装
  17. WinForm应用程序中实现自动更新功能
  18. jquery相对定位(包含find的使用 find相当于后代选择器)$(&quot;选择器1&quot;,&quot;选择器2&quot;)
  19. 辣鸡蒟蒻Klaier的一些计划
  20. TopcoderSRM679 Div1 250 FiringEmployees(树形dp)

热门文章

  1. Delphi单元文件之-简体繁体互转
  2. 数据流重定向和管道命令, grep, tr,sort, wc, cut,split,tee,sleep(shell 02)
  3. profile MySQL性能分析工具
  4. POJ 2823 Sliding Window(单调队列入门题)
  5. Spring XML和Annotation混合配置的时候,XML中Bean名称写错会导致启动异常不打印、死循环
  6. 转:Oracle里几组重要的视图--v$sysstat,v$system_event,v$parameter v$system_parameter
  7. SecureCRT乱码问题解决方法
  8. 爬取github上流行的python项目
  9. curl 命令参数
  10. flask系列三之Jinja2模板