异步 Action 的定义
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);
});
}
}
异步 Action 的定义的更多相关文章
- ASP.NET MVC下的异步Action的定义和执行原理
一.基于线程池的请求处理ASP.NET通过线程池的机制处理并发的HTTP请求.一个Web应用内部维护着一个线程池,当探测到抵达的针对本应用的请求时,会从池中获取一个空闲的线程来处理该请求.当处理完毕, ...
- ASP.NET MVC下的异步Action的定义和执行原理[转]
http://www.cnblogs.com/artech/archive/2012/06/20/async-action-in-mvc.html Visual Studio提供的Controller ...
- 夜深了 关于 异步Action的定义的截图
- Redux 处理异步 Action
redux-promise-utils What redux-promise-utils 是一个基于 redux-thunk 和 redux-actions 的工具,符合 FSA 规范,方便开发者处理 ...
- redux-amrc:用更少的代码发起异步 action
很多人说 Redux 代码多,开发效率低.其实 Redux 是可以灵活使用以及拓展的,经过充分定制的 Redux 其实写不了几行代码.今天先介绍一个很好用的 Redux 拓展-- redux-amrc ...
- redux学习日志:关于异步action
当我们在执行某个动作的时候,会直接dispatch(action),此时state会立即更新,但是如果这个动作是个异步的呢,我们要等结果出来了才能知道要更新什么样的state(比如ajax请求),那就 ...
- ASP.NET MVC什么时候使用异步Action
在没有使用异步Action之前,在Action内,比如有如下的写法: public ActionResult Index() CustomerHelper cHelper = new Customer ...
- 【React全家桶入门之十三】Redux中间件与异步action
在上一篇中我们了解到,更新Redux中状态的流程是这种:action -> reducer -> new state. 文中也讲到.action是一个普通的javascript对象.red ...
- Redux学习笔记--异步Action和Middleware
异步Action 之前介绍的都是同步操作,Redux通过分发action处理state,所有的数据流都是同步的,如果需要一步的话怎么办? 最简单的方式就是使用同步的方式来异步,将原来同步时一个acti ...
随机推荐
- Java 泛型分析
Java 泛型 Java Generics 是 JDK 1.5 引入的新特性,它提供了编译时的类型安全检测机制,避免了代码中进行显示的类型转换(cast),是对类型系统的一种增强. Java Gene ...
- xunsearch使用SCWS
目录 创建分词对象 获取分词结果 提取重要词汇 在 Xunsearch 使用 SCWS 创建分词对象 $xs = new XS('my'); // 必须先创建一个 xs 实例,否则会抛出异常 $tok ...
- JavaScript笔记——事件
事件一般是用于浏览器和用户操作进行交互.最早是 IE 和 Netscape Navigator 中出现, 作为分担服务器端运算负载的一种手段.直到几乎所有的浏览器都支持事件处理.而 DOM2 级规范开 ...
- spring boot 集成 rabbitmq
1.使用默认的AmqpTemplate生产消费pojo时,pojo需要implement Serializable,否则会抛出org.springframework.amqp.AmqpExceptio ...
- Linq入门博客系列地址http://www.cnblogs.com/lifepoem/category/330218.html
http://www.cnblogs.com/lifepoem/category/330218.html Linq及LambdaSql语句: http://kb.cnblogs.com/page/42 ...
- leetcode700
class Solution { public: TreeNode* searchBST(TreeNode* root, int val) { if (root == NULL) { return n ...
- 虚拟机之 搭建discuz论坛
1.下载 mkdir /data/www cd !$ wget http://download.comsenz.com/DiscuzX/3.2/Discuz_X3.2_SC_GBK.zip 2.解压 ...
- Hadoop Serialization -- hadoop序列化详解 (3)【ObjectWritable,集合Writable以及自定义的Writable】
前瞻:本文介绍ObjectWritable,集合Writable以及自定义的Writable TextPair 回顾: 前面了解到hadoop本身支持java的基本类型的序列化,并且提供相应的包装实现 ...
- Timer中的重要函数
- 安卓 textview 换行 不满就换了
public static String ToDBC(String input) { char[] c = input.toCharArray(); for (int i = 0; i < c. ...