每个请求来到Web容器,Web容器会为其分配一条执行绪来专门负责该请求,直到回应完成前,该执行绪都不会被释放回容器。 执行绪会耗用系统资源,若有些请求需要长时间处理(例如长时间运算、等待某个资源),就会长时间占用执行绪,若这类的请求很多,许多执行绪都被长时间占用,对于系统就会是个效能负担,甚至造成应用程式的效能瓶颈。

基本上一些需长时间处理的请求,通常客户端也较不在乎请求后要有立即的回应,若可以,让这类请求先释放容器分配给该请求的执行绪,让容器可以有机会将执行绪资源分配给其它的请求,可以减轻系统负担。 原先释放了容器所分配执行绪的请求,其回应将被延后,直到处理完成(例如长时间运算完成、所需资源已获得)再行对客户端的回应。

在Servlet 3.0中,在ServletRequest上提供了 startAsync( )方法:

  1. AsyncContext startAsync() throws java.lang.IllegalStateException;
  2. AsyncContext startAsync(ServletRequest servletRequest,
  3. ServletResponse servletResponse)
  4. throws java.lang.IllegalStateException

这两个方法都会传回AsyncContext介面的实作物件,前者会直接利用原有的请求与回应物件来建立AsyncContext ,后者可以让你传入自己建立的请求、回应包裹物件。 在呼叫了startAsync()方法取得AsyncContext物件之后,这次的回应会被延后,并释放容器所分配的执行绪。

你可以透过AsyncContext的getRequest() 、 getResponse()方法取得请求、回应物件,此次对客户端的回应将暂缓至呼叫AsyncContext的complete()方法或dispatch()为止,前者表示回应完成,后者表示将回应调派给指定的URL 。

若要能呼叫ServletRequest的startAsync()使用AsyncContext,你的 Servlet 必须能支援非同步处理,如果使用@WebServlet来标注,则可以设定其asyncSupported为true 。 例如:

  1. @WebServlet(urlPatterns = "/some.do", asyncSupported = true )
  2. public class AsyncServlet extends HttpServlet {
  3. ...

如果使用web.xml设定Servlet,则可以设定<async-supported>标签为true :

  1. ...
  2. <servlet>
  3. <servlet-name>AsyncServlet</servlet-name>
  4. <servlet-class>cc.openhome.AsyncServlet</servlet-class>
  5. <async-supported>true</async-supported>
  6. </servlet>
  7. ...

如果Servlet将会非同步处理,若其前端有过滤器,则过滤器亦需标示其支援非同步处理,如果使用@WebFilter ,同样是可以设定其asyncSupported为true 。 例如:

  1. @WebFilter(urlPatterns = "/some.do", asyncSupported = true )
  2. public class AsyncFilter implements Filter{
  3. ...

如果使用web.xml设定过滤器,则可以设定<async-supported>标签为true :

  1. ...
  2. <filter>
  3. < filter -name>AsyncFilter</ filter -name>
  4. < filter -class>cc.openhome.AsyncFilter</ filter -class>
  5. <async-supported>true</async-supported>
  6. </ filter >
  7. ...

底下示范一个非同步处理的例子,对于进来的请求,Servlet会取得其AsyncContext ,并释放容器所分配的执行绪,回应被延后,对于这些被延后回应的请求,建立一个Runnable的物件,并将其排入一个执行绪池(Thread pool),执行绪池的执行绪数量是固定的,让这些必须长时间处理的请求,在这些有限数量的执行绪中完成,而不用每次请求都占用容器所分配的执行绪。

  1. package cc.openhome;
  2. import java.io.*;
  3. import java.util.concurrent.*;
  4. import javax.servlet.*;
  5. import javax.servlet.annotation.*;
  6. import javax.servlet.http.*;
  7. @WebServlet(name="AsyncServlet", urlPatterns={"/async.do"},
  8. asyncSupported = true )
  9. public class AsyncServlet extends HttpServlet {
  10. // 执行绪池
  11. private ExecutorService executorService = Executors.newFixedThreadPool(10);
  12. protected void processRequest(HttpServletRequest request,
  13. HttpServletResponse response)
  14. throws ServletException, IOException {
  15. response.setContentType("text/html; charset=UTF8");
  16. AsyncContext ctx = request.startAsync();
  17. executorService.submit(new AsyncRequest(ctx));
  18. }
  19. @Override
  20. protected void doGet(HttpServletRequest request,
  21. HttpServletResponse response)
  22. throws ServletException, IOException {
  23. processRequest(request, response);
  24. }
  25. @Override
  26. protected void doPost(HttpServletRequest request,
  27. HttpServletResponse response)
  28. throws ServletException, IOException {
  29. processRequest(request, response);
  30. }
  31. @Override
  32. public void destroy() {
  33. executorService.shutdown();
  34. }
  35. }

AsyncRequest是个实作Runnable的类别,其模拟了长时间处理:

    1. package cc.openhome;
    2. import java.io.PrintWriter;
    3. import javax.servlet.AsyncContext;
    4. public class AsyncRequest implements Runnable {
    5. private AsyncContext ctx;
    6. public AsyncRequest( AsyncContext ctx ) {
    7. this.ctx = ctx;
    8. }
    9. @Override
    10. public void run() {
    11. try {
    12. // 模拟长时间的处理
    13. Thread.sleep(10000);
    14. PrintWriter out = ctx.getResponse() .getWriter();
    15. out.println("久等了...XD");
    16. // 这边才真正送出回应
    17. ctx.complete();
    18. } catch (Exception e) {
    19. e.printStackTrace();
    20. }
    21. }
    22. }  

Servlet3.0: 简介AsyncContext的更多相关文章

  1. Spring注解开发系列VII --- Servlet3.0

    Servlet3.0简介 Servlet 3.0 作为 Java EE 6 规范体系中一员,随着 Java EE 6 规范一起发布.该版本在前一版本(Servlet 2.5)的基础上提供了若干新特性用 ...

  2. Servlet3.0的异步

    servlet之前的操作同时同步的,就是按照这样的一个流程来走的: 1.请求根据一个路径路由到一个servlet中, 2.servlet获取一系列的参数 3.执行一系列的逻辑(花费时间所占的比重也更大 ...

  3. Servlet3.0中Servlet的使用

    目录 1.注解配置 2.异步调用 3.文件上传 相对于之前的版本,Servlet3.0中的Servlet有以下改进: l  支持注解配置. l  支持异步调用. l  直接有对文件上传的支持. 在这篇 ...

  4. Servlet3.0新特性

    1 Servlet3.0新特性概述 使用要求:MyEclipse10.0或以上版本,发布到Tomcat7.0或以上版本,创建JavaEE6.0应用! Servlete3.0的主要新特性如下三部分: 使 ...

  5. Servlet3.0的新特性

    注意:Servlet3.0的项目一定要使用Tomcat7.0才能看到效果!! 1.新增标注支持     在Servlet3.0的部署描述文件web.xml的顶层标签<web-app>中有一 ...

  6. [译]servlet3.0与non-blocking服务端推送技术

    Non-blocking(NIO)Server Push and Servlet 3 在我的前一篇文章写道如何期待成熟的使用node.js.假定有一个框架,基于该框架,开发者只需要定义协议及相关的ha ...

  7. Servlet3.0新特性使用详解

    可插拔的Web框架 几乎所有基于Java的web框架都建立在servlet之上.现今大多数web框架要么通过servlet.要么通过Web.xml插入.利用标注(Annotation)来定义servl ...

  8. Servlet3.0新特性(从注解配置到websocket编程)

    Servlet3.0的出现是servlet史上最大的变革,其中的许多新特性大大的简化了web应用的开发,为广大劳苦的程序员减轻了压力,提高了web开发的效率.主要新特性有以下几个: 引入注解配置 支持 ...

  9. Java自学手记——servlet3.0新特性

    servlet3.0出来已经很久了,但市场上尚未普遍应用,servlet3.0有三个比较重要的新特性:使用注解来代替配置文件,异步处理以及上传组件支持. 支持servlet3.0的要求:MyEclip ...

随机推荐

  1. [HNOI2018]道路(DP)

    题目描述 W 国的交通呈一棵树的形状.W 国一共有n−1n - 1n−1 个城市和nnn 个乡村,其中城市从111 到n−1n - 1n−1 编号,乡村从111 到nnn 编号,且111 号城市是首都 ...

  2. [Contest20180328]同构

    我们先把两棵树的所有节点的儿子排序,然后dfs,记录访问的深度序列 然后可以发现题目中的两个操作都只会在深度序列中删掉一位,不会对其他位有影响(自己画画图就知道了) 对于一个深度序列$d_{1\cdo ...

  3. 关于Hadoop_env.sh中的HADOOP_CLASSPATH

    之前博客里介绍了如何自定义DoubleArrayWritable,并将该类型的value写入SequenceFile文件中,为了能够使用命令查看这个文件中的内容(果然坑都是一步一步给自己挖的)参考了网 ...

  4. 【棋盘DP】【OpenJudge7614】最低通行费

    最低通行费 总时间限制: 1000ms 内存限制: 65536kB [描述] 一个商人穿过一个 N*N 的正方形的网格,去参加一个非常重要的商务活动.他要从网格的左上角进,右下角出.每穿越中间1个小方 ...

  5. httpclient4.3访问https

    1.创建一个访问https的工具类 package org.aaa.tool;import java.io.File; import java.io.IOException; import java. ...

  6. iOS 修改APP工程名字

    我们在iOS开发中,难免会遇到项目做到一半要改名字的情况.如果项目名差的太大,工程名看起来总是不舒服的,就会想着为工程改个贴切的名字,那么你就为用到本文记录的内容. 如何修改工程名呢? 下面我就拿一个 ...

  7. 【重装系统】老毛桃U盘工具V2013超级装机版-安装原版Win7/Win8

    老毛桃U盘工具V2013超级装机版-程序下载和运行 老毛桃U盘工具V2013超级装机版-安装原版XP的方法 老毛桃U盘工具V2013超级装机版-安装原版Win7/Win8

  8. Nginx流量带宽请求状态统计(ngx_req_status)

    介绍           ngx_req_status 用来展示 nginx 请求状态信息,类似于 apache 的 status, nginx 自带的模块只能显示连接数等等 信息,我们并不能知道到底 ...

  9. Android API level 版本对应关系

    详情地址:http://developer.android.com/guide/topics/manifest/uses-sdk-element.html Platform Version API L ...

  10. linux基础-第十九单元_nfs服务

    #服务端部署 介绍: NFS 是Network File System的缩写,即网络文件系统.一种使用于分散式文件系统的协定,由Sun公司开发,于1984年向外公布.功能是通过网络让不同的机器.不同的 ...