`

Servlet3新特性学习笔记——异步处理支持

 
阅读更多

1、传统Servlet处理

Web容器会为每个请求分配一个线程,默认情况下,响应完成前,该线程占用的资源都不会被释放。若有些请求需要长时间(例如长处理时间运算、等待某个资源),就会长时间占用线程所需资源,若这类请求很多,许多线程资源都被长时间占用,会对系统的性能造成负担。

 

 

 

2、新特性:异步处理

Servlet 3.0新增了异步处理,可以先释放容器分配给请求的线程与相关资源,减轻系统负担,原先释放了容器所分配线程的请求,其响应将被延后,可以在处理完成(例如长时间运算完成、所需资源已获得)时再对客户端进行响应。

 

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

  • 第一步,Servlet 接收到请求之后,可能需要对请求携带的数据进行一些预处理;
  • 第二步,调用业务接口的某些方法,以完成业务处理;
  • 第三步,根据处理的结果提交响应,Servlet 线程结束。

其中第二步的业务处理通常是最耗时的,这主要体现在数据库操作,以及其它的跨网络调用等,在此过程中,Servlet 线程一直处于阻塞状态,直到业务方法执行完毕。在处理业务的过程中,Servlet 资源一直被占用而得不到释放,对于并发较大的应用,这有可能造成性能的瓶颈。对此,在以前通常是采用私有解决方案来提前结束 Servlet 线程,并及时释放资源。

 

 

Servlet 3.0 针对这个问题做了开创性的工作,现在通过使用 Servlet 3.0 的异步处理支持,之前的 Servlet 处理流程可以调整为如下的过程:

  • 第一步,Servlet 接收到请求之后,可能首先需要对请求携带的数据进行一些预处理;
  • 第二步,Servlet 线程将请求转交给一个异步线程来执行业务处理,线程本身返回至容器,
  • 第三步,Servlet 还没有生成响应数据,异步线程处理完业务以后,可以直接生成响应数据(异步线程拥有 ServletRequest 和 ServletResponse 对象的引用),或者将请求继续转发给其它 Servlet。

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

 

 

 

3、配置

异步处理特性可以应用于 Servlet 和Filter两种组件,由于异步处理的工作模式和普通工作模式在实现上有着本质的区别,因此默认情况下,Servlet 和Filter并没有开启异步处理特性,若需要使用,需按照如下的方式启用:

方式一:

对于使用传统的部署描述文件 (web.xml) 配置 Servlet 和过滤器的情况,Servlet 3.0 为 <servlet> 和 <filter> 标签增加了 <async-supported> 子标签,该标签的默认取值为 false,要启用异步处理支持,则将其设为 true 即可。以 Servlet 为例,其配置方式如下所示:

<servlet> 
    <servlet-name>DemoServlet</servlet-name> 
    <servlet-class>footmark.servlet.Demo Servlet</servlet-class> 
    <async-supported>true</async-supported> 
</servlet>

 

方式二:

对于使用 Servlet 3.0 提供的 @WebServlet 和 @WebFilter 进行 Servlet 或过滤器配置的情况,这两个注解都提供了 asyncSupported 属性,默认该属性的取值为 false,要启用异步处理支持,只需将该属性设置为 true 即可。以 @WebFilter 为例,其配置方式如下所示:

@WebFilter(urlPatterns = "/demo",asyncSupported = true) 
public class DemoFilter implements Filter{...} 

 

 

 

4、AsyncContext

Servlet3的异步处理是通过AsyncContext来处理的。AsyncContext对象代表异步处理的上下文,它提供一些工具方法,可完成设置异步调用的超时时长,dispatch用于请求、启动后台线程、获取request、response对象等功能。

 

Servlet可以通过ServletRequest的如下两个方法开启异步调用、创建AsyncContext对象:

AsyncContext startAsync()

AsyncContext startAsync(ServletRequest , ServletRespone)

 

重复调用以上的方法得到同一个AsyncContext对象。这两个方法都会返回AsyncContext接口的实现对象,前者会直接利用原有的请求与响应对象来创建AsyncContext,后者可以传入自行创建的请求、响应封装对象。在调用了startAsync()方法取得AsyncContext对象之后,此次请求的响应会被延后,并释放容器分配的线程。

 

可以通过AsyncContext的getRequest()、getResponse()方法取得请求、响应对象,此次对客户端的响应将暂缓至调用AsyncContext的complete()或dispatch()方法为止,前者表示响应完成,后者表示将调派指定的URL进行响应。

 

 

一个简单的模拟异步处理的 Servlet 示例如下:

@WebServlet(urlPatterns = "/demo", asyncSupported = true)
public class AsyncDemoServlet extends HttpServlet {
    @Override
    public void doGet(HttpServletRequest req, HttpServletResponse resp)
    throws IOException, ServletException {
        resp.setContentType("text/html;charset=UTF-8");
        PrintWriter out = resp.getWriter();
        out.println("进入Servlet的时间:" + new Date() + ".");
        out.flush();
 
        //在子线程中执行业务调用,并由其负责输出响应,主线程退出
        AsyncContext ctx = req.startAsync();
        new Thread(new Executor(ctx)).start();
 
        out.println("结束Servlet的时间:" + new Date() + ".");
        out.flush();
    }
}
 
/**
* 线程类
*/
public class Executor implements Runnable {
    private AsyncContext ctx = null;
    public Executor(AsyncContext ctx){
        this.ctx = ctx;
    }
 
    public void run(){
        try {
            //等待十秒钟,以模拟业务方法的执行
            Thread.sleep(10000);
            PrintWriter out = ctx.getResponse().getWriter();
            out.println("业务处理完毕的时间:" + new Date() + ".");
            out.flush();
            ctx.complete();   //完成异步调用。
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

 

 

 

5、监听器 AsyncListener

当Servlet启用异步调用的线程之后,该线程的执行过程对开发者是透明的。但在有些情况下,开发者需要了解该异步线程的执行细节,并针对特定的执行结果进行针对性处理。Servlet 3为异步处理提供了一个监听器,使用 AsyncListener 接口表示。它可以监控如下四种事件:

  • 异步线程开始时,调用 AsyncListener 的 onStartAsync(AsyncEvent event) 方法。
  • 异步线程出错时,调用 AsyncListener 的 onError(AsyncEvent event) 方法。
  • 异步线程执行超时,调用 AsyncListener 的 onTimeout(AsyncEvent event) 方法。
  • 异步执行完毕时,调用 AsyncListener 的 onComplete(AsyncEvent event) 方法。

要注册一个 AsyncListener,只需将准备好的 AsyncListener 对象传递给 AsyncContext 对象的 addListener() 方法即可,如下所示:

AsyncContext ctx = req.startAsync(); 
ctx.addListener(new AsyncListener() { 
    public void onComplete(AsyncEvent asyncEvent) throws IOException { 
        // 做一些清理工作或者其他
    } 
    ... 
}); 

 

 

分享到:
评论

相关推荐

    servlet和jsp学习笔记

    这篇学习笔记将深入探讨这两个概念,以及它们在实际开发中的应用。 Servlet是Java编程语言的一个接口,由Sun Microsystems(现为Oracle)定义,它允许Java代码与HTTP服务器交互。Servlet主要负责处理客户端的请求并...

    servlet学习笔记

    【Servlet学习笔记】是关于Java Web开发中一个关键组件——Servlet的详细教程,它主要讲述了前端与后端如何协同工作,实现数据的交互和页面的动态呈现。在Web应用程序开发中,Servlet扮演着服务器端程序的角色,处理...

    DWR 2.0M3 学习笔记

    3. **使用DWR接口**:DWR提供了两种主要的接口——`Call`和`Batch`。`Call`接口用于单个方法调用,而`Batch`接口则允许一次调用多个方法,提高了效率。通过这些接口,JavaScript可以直接调用Java对象的方法,就像...

    java私人学习笔记

    Java私人学习笔记主要涵盖了Java编程语言以及与其相关的几个著名框架——Spring、Struts2和Hibernate,还有前端开发中常用的JavaScript。这些技术在IT行业中的应用广泛,是许多企业级应用开发的基础。以下是对这些...

    java学习笔记

    ### Java学习笔记知识点详解 #### Java 平台与开发概述 - **Java平台分类:** - **J2SE(Java SE):** 标准版,适用于桌面应用程序开发。 - **J2EE(Java EE):** 企业版,专为Web应用和服务端应用程序设计。 -...

    学习笔记 MHT2

    【学习笔记 MHT2】是一份综合性的IT学习资料,主要涵盖了JavaScript编程、软件开发工具、Web服务器技术、编码解码方法以及设计模式等多个领域的知识。以下是对这些知识点的详细阐述: 1. **JavaScript经典技巧**:...

    ActiveMQ学习笔记之一--ActiveMQ下载

    **ActiveMQ学习笔记之一——ActiveMQ下载** ActiveMQ是Apache软件基金会开发的一个开源消息中间件,它是基于Java消息服务(JMS)规范的,用于在分布式系统中传递消息。作为一个高性能、可伸缩且可靠的解决方案,...

    javaweb笔记+狂神+知识总结

    "狂神"可能是指一位知名的JavaWeb讲师或教程作者,他的笔记为学习者提供了深入理解和实践该技术的宝贵资源。下面将详细阐述JavaWeb的主要知识点。 1. **Servlet**:Servlet是Java Web应用程序的核心组件,它是一个...

    51CTO下载-韩顺平java从入门到精通视频教程(全94讲)学习笔记整理(齐全)

    它包含了 J2SE 的所有特性,并额外提供了一些高级服务,如分布式组件模型、Web 服务支持等。 - **特点**:能够处理高负载、高并发的企业级应用需求。 - **应用场景**:适用于大型网站、电子商务平台等企业级项目。 ...

    MLDN_J2EE框架_笔记

    **J2EE框架详解——基于MLDN的学习笔记** J2EE(Java 2 Platform, Enterprise Edition)是Java平台上用于构建企业级应用的框架,它提供了一系列的组件和服务,旨在简化多层分布式应用的开发和部署。这套“MLDN_J2EE...

    完善后的在线云笔记系统

    《完善后的在线云笔记系统——基于mybatis、javaee、ajax和json的技术实践》 在线云笔记系统在当今数字化时代中扮演着重要角色,它为用户提供了一种方便、快捷的记录、整理和共享信息的方式。本项目是针对一个已有...

    崔希凡javaweb笔记day08-day13

    崔希凡老师的javaweb笔记涵盖了从day08到day13的重要学习内容,这是传智播客官网上一套详尽的视频教程的配套资料。这些笔记和源码旨在帮助学员深入理解JavaWeb开发的核心技术,提升编程技能。以下是根据这些天的笔记...

    Java毕业设计——基于ssm的兼职论坛设计与实现(源码+数据库).zip

    - **Spring**:核心是依赖注入(DI),用于管理对象的生命周期和装配,同时也提供AOP(面向切面编程)支持,用于处理如日志、事务等通用逻辑。 - **SpringMVC**:是Spring的一个模块,负责Web层的请求分发和处理,...

    JQuery.rar_Jsp/Servlet_Java_

    **jQuery——轻量级的JavaScript库** jQuery是一个高效、易用的JavaScript库,它极大地简化了JavaScript编程,尤其是在处理HTML文档、事件处理、动画效果以及AJAX交互方面。标题中的"JQuery.rar"是一个包含jQuery...

Global site tag (gtag.js) - Google Analytics