servlet3.0异步化简析

tomcat处理请求流程

从客户端发出一次请求至tomcat容器大致经过如下过程:

  1. 客户端发出http请求至tomcat的连接监听端口;
  2. tomcat connector接收线程接收请求,并根据http协议解析该次请求;
  3. tomcat 通过解析的http报文,初始化org.apache.coyote.Request,并实例化org.apache.coyote.Response;
  4. 经装饰模式转化为servlet api对应的HttpServletRequest与HttpServletReponse;
  5. 经tomcat的层层容器engine,host,context最终到过我们所写的业务servlet的service方法;
  6. 业务方法service,处理相关的业务逻辑,写入相应的响应的至response,并返回tomat的容器组件;
  7. tomcat该处理线程关闭响应流Response并将响应内容返回客户端;
  8. tomcat该处理线程被释放,然后用于下次请求的处理;

从中可以看出tomcat服务器在处理请求时在servlet中只有在相应的处理方法执行结束后才会将响应数据发送至客户端并关闭线程释放资源。所以如果在处理请求时如果处理的代码时间过程对当前线程也会一直占用,那么是否存在一种类似于ajax的技术,使当前servlet服务先继续向下执行,然后等请求执行玩动态返回结果,这就是servlet3的异步化,当然现在基本上还是使用ajax来实现请求的异步

从上面tomcat处理请求过程可知, tomcat处理线程一直被占用,直至业务方法处理完毕;那么servlet3是如何进行异步化的呢?使用servlet3的异步的一般姿势:

  AsyncContext asyncContext = request.startAsync();//获取异步上下文对象
  asyncContext.start(new IndexThread(asyncContext));//创建线程执行异步线程

  asyncContext.complete();异步线程执行完通知主线程执行结束

当然在使用上面的代码之前,不要忘了开启servlet的异步化(使用注解也可以asyncSupported = true):

<servlet>
    <servlet-name>test</servlet-name>
    <servlet-class>***</servlet-class>
    <async-supported>true</async-supported>
</servlet>

 

代码实例

TestServlet:执行异步化的servlet

import java.io.IOException;
import java.io.PrintWriter;
import javax.servlet.AsyncContext;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
@WebServlet(urlPatterns = "/test",asyncSupported = true)
public class TestServlet extends HttpServlet {
	private static final long serialVersionUID = 1L;
	protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
		response.setContentType("text/html;charset=utf-8");
	        try {
	            PrintWriter out = response.getWriter();
	            out.println("servlet started.<br/>");
	            out.flush();
	            AsyncContext asyncContext = request.startAsync();
	            asyncContext.start(new IndexThread(asyncContext));
	            out.println("servlet end.<br/>");
	            out.flush();
	        } catch (Exception e) {
	            e.printStackTrace();
	        }
	    }
}

indexThread:异步执行的线程

import java.io.PrintWriter;
import javax.servlet.AsyncContext;
public class IndexThread implements Runnable{
	private AsyncContext asyncContext;
    public IndexThread(AsyncContext asyncContext) {
        this.asyncContext = asyncContext;
    }
    public void run() {
        try {
            Thread.sleep(5000);
            PrintWriter out = asyncContext.getResponse().getWriter();
            out.println("hello servlet3.<br/>");
            out.flush();
            asyncContext.complete();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

 

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值