Servlet3.0新特性-异步处理Servlet、Filter

一、浅谈Servlet 在Servlet 3.0 之前,一个普通 Servlet 的主要工作流程大致如下:

其中黄色阶段通常是最耗时的,因为业务处理一般涉及数据库操作,还会受到网络等的影响,而在此过程中,Servlet 线程一直处于阻塞状态,直到业务处理完毕。

在处理业务的过程中,Servlet 资源一直被占用而得不到释放,对于并发较大的应用,这有可能造成性能的瓶颈。对此,在以前通常是采用私有解决方案来提前结束 Servlet 线程,并及时释放资源。

为解决这个问题,Servlet 3.0 就开始支持异步处理了,这与 Ajax 异步不一样,之前的 Servlet 处理流程可以调整为如下的过程:

首先,Servlet 接收到请求之后,可能首先需要对请求携带的数据进行一些预处理;

接着,Servlet 线程将请求转交给一个异步线程来执行业务处理,线程本身返回至容器,

此时 Servlet 还没有生成响应数据,异步线程处理完业务以后,可以直接生成响应数据(异步线程拥有 ServletRequest 和 ServletResponse 对象的引用),或者将请求继续转发给其它 Servlet。

如此一来, Servlet 线程不再是一直处于阻塞状态以等待业务逻辑的处理,而是启动异步线程之后可以立即返回。

二、启用异步处理 异步处理特性可以应用于 Servlet 和过滤器两种组件。

在默认情况下,Servlet 和过滤器并没有开启异步处理特性,因为异步处理的工作模式和普通工作模式在实现上有着本质的区别,因此如果希望使用该特性,则必须按照如下的方式启用:

1. 在 web.xml 文件中启动 Servlet3.0 默认是没有 web.xml 文件的,但 Servlet3.0 也是支持 web.xml 文件的,较 Servlet之前的版本,Servlet 3.0 在 和 标签中增加了 子标签,该标签默认是 false 。如果想启用异步支持,只需要置为 true 即可。

例如: asynServlet com.servlet.AsynServlet true asynFilter com.filter.AsynFilter true 1 2 3 4 5 6 7 8 9 10 11 12 2. 通过注解开启异步支持 Servlet 3.0 提供的 @WebServlet 和 @WebFilter 进行 Servlet 或 Filter 配置的情况,这两个注解都提供了 asyncSupported 属性,默认该属性的取值为 false。

如果想启用异步支持,只需要置为 true 即可。

例如: @WebServlet(value="/asyn-servlet",asyncSupported=true) public class ServletAsyn extends HttpServlet {...} @WebFilter(value="/*",asyncSupported=true) public class FilterAsyn implements Filter {...} 1 2 3 4 5

三、异步处理的代码实现方式 Servlet 和 Filter 使用异步支持的实现方式是一模一样的操作。这里只以 Servlet 的实现方式为例。

1. 模拟注册发信息到邮件的简单的异步处理

Servlet 示例代码:

package com.servlet; import java.io.IOException; 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(value="/asyn-servlet",asyncSupported=true) public class ServletAsyn extends HttpServlet { private static final long serialVersionUID = 1L; protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { System.out.println("Servlet is start");

//1.获得异步上下文对象 AsyncContext ac = request.startAsync();

//2.启动一个耗时的子线程 ThreadTask tt = new ThreadTask(ac);

//3.可设置异步超时对象,需在启动异步上下文对象前设置 /* * 设置超时后,在超时时间内子线程没有结束,主线程则会停止等待,继续往下执行 */ ac.setTimeout(3000);

//4.开启异步上下文对象 ac.start(tt);

//主线程结束向客户端发送消息 System.out.println("Servlet is end"); response.setContentType("text/html;charset=utf-8"); response.getWriter().append("信息已发送到邮箱"); } protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doGet(request, response); } } 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43

异步线程的执行类:

package com.servlet; import java.util.Date; import javax.servlet.AsyncContext; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpSession; public class ThreadTask implements Runnable{ private AsyncContext ac;

//定义一个异步上下文 public ThreadTask(AsyncContext ac) { super(); this.ac = ac; } @Override public void run() { /* * 服务端异步典型应用是注册时向邮箱发送验证码 */ try { //进行异步的一些处理 HttpServletRequest requst = (HttpServletRequest) ac.getRequest(); HttpSession session = requst.getSession(); System.out.println("asyn-task start" + new Date()); for(int i=5;i>0; i--) { System.out.println(i); Thread.sleep(1000); }

//将结果放到session等方式 session.setAttribute("message", "This is the result of asyn"); System.out.println("asyn-task end" + new Date());

//通知主线程已经处理完成 /* * 除了使用 ac.complete() 方法通知主线程已经处理外 * 还可以使用 ac.dispatch() 方法重定向到一个页面 */ ac.dispatch("/show.jsp"); } catch (Exception e) { e.printStackTrace(); } } } 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 这里的 show.jsp 模拟为指定邮箱。show.jsp页面body标签内代码为:

邮箱内容:${message }

2. 模拟注册发信息到邮件的简单的异步处理 Servlet 测试过程 先访问 show.jsp 页面,结果如下:

然后我们访问servlet:http://localhost:8080/Test2/asyn-servlet,在页面显示的是 然后我们看一下控制台打印的结果:

最后我们再回过去看一下 show.jsp 页面

三、对异步处理过程的监听 除此之外,Servlet 3.0 还为异步处理提供了一个监听器,使用 AsyncListener 接口表示。

它可以监控如下四种事件:

(1)异步线程开始时,调用 AsyncListener 的 onStartAsync(AsyncEvent event) 方法;

(2)异步线程出错时,调用 AsyncListener 的 onError(AsyncEvent event) 方法;

(3)异步线程执行超时,则调用 AsyncListener 的 onTimeout(AsyncEvent event) 方法;

(4)异步执行完毕时,调用 AsyncListener 的 onComplete(AsyncEvent event) 方法。 如果要注册一个 AsyncListener,只需将准备好的 AsyncListener 对象传递给 AsyncContext 对象的 addListener() 方法即可,如下所示:

AsyncContext ac = request.startAsync(); ac.addListener(new AsyncListener() { @Override public void onComplete(AsyncEvent arg0) throws IOException { // TODO Auto-generated method stub } @Override public void onError(AsyncEvent arg0) throws IOException { // TODO Auto-generated method stub } @Override public void onStartAsync(AsyncEvent arg0) throws IOException { // TODO Auto-generated method stub } @Override public void onTimeout(AsyncEvent arg0) throws IOException { // TODO Auto-generated method stub } });

点赞(0) 打赏

评论列表 共有 0 条评论

暂无评论

热门产品

php编程基础教程.pptx|php编程培训,php,编程,基础,教程,pptx
php编程基础教程.pptx

历史上的今天:04月19日

热门专题

自考本科|自考本科有用吗,自考文凭,自考本科文凭,自考文凭有用吗,自考本科文凭有用吗,自考文凭承认吗
自考本科
易捷尔单招|易捷尔单招,易捷尔单招培训,易捷尔单招报名,易捷尔单招考试,易捷尔单招培训学校,易捷尔单招分数
易捷尔单招
卓越综合高中|卓越综合高中
卓越综合高中
弥勒综合高中|弥勒综合高中
弥勒综合高中
云南开放大学|云南开放大学报名,云南开放大学报考,云南开放大学,什么是云南开放大学,云南开放大学学历,云南开放大学学费,云南开放大学报名条件,云南开放大学报名时间,云南开放大学学历,云南开放大学专业
云南开放大学
昆明综合高中|昆明综合高中
昆明综合高中
APP开发|app开发_app开发公司_app软件开发_专业app开发_云南app开发公司_app定制_原生app开发定制
APP开发
大理科技管理学校|大理科技管理学校,大理科技,大理科技中等职业技术学校,大理科技管理中等职业技术学校,大理科技学校
大理科技管理学校

微信小程序

微信扫一扫体验

立即
投稿

微信公众账号

微信扫一扫加关注

发表
评论
返回
顶部