Tomcat怎么实现异步Servlet

 更新时间:2017年01月03日 10:30:20   作者:seaboat 汪洋之舟  
这篇文章主要介绍了Tomcat怎么实现异步Servlet,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧。

有时Servlet在生成响应报文前必须等待某些耗时的操作,比如在等待一个可用的JDBC连接或等待一个远程Web服务的响应。对于这种情况servlet规范中定义了异步处理方式,由于Servlet中等待阻塞会导致Web容器整体的处理能力低下,所以对于比较耗时的操作可以放置到另外一个线程中进行处理,此过程保留连接的请求和响应对象,在处理完成之后可以把处理的结果通知到客户端。

下面先看Servlet在同步情况下的处理过程,如图所示,Tomcat的客户端请求由管道处理最后会通过Wrapper容器的管道,这时它会调Servlet实例的service方法进行逻辑处理,处理完后响应客户端,整个处理由Tomcat的Executor线程池的线程处理,而线程池的最大线程数使有限制的,所以这个处理过程越短、越快把线程让回线程池就越好。但如果Servlet中的处理逻辑耗时越长就会导致长期地占用Tomcat的处理线程池,影响Tomcat的整体处理能力。

 Servlet同步处理

为了解决上面的问题引入了支持异步的Servlet,同样是客户端请求到来,然后通过管道最后进入到Wrapper容器的管道,调用Servlet实例的service后,创建一个异步上下文将耗时的逻辑操作封装起来,交给用户自己定义的线程池,这时Tomcat的处理线程就能马上回到Executor线程池,而不用等待耗时的操作完成才让出线程,从而提升了Tomcat的整体处理能力。这里要注意的是,由于后面做完耗时的操作后还需要对客户端响应,所以需要保持住Request和Response对象,以便输出响应报文到客户端。

Servlet异步处理

再结合一个简单的异步代码来看Tomcat对Servlet异步的实现:

public class AsyncServlet extends HttpServlet {

  ScheduledThreadPoolExecutor userExecutor = new ScheduledThreadPoolExecutor(5);

  public void doGet(HttpServletRequest req, HttpServletResponse res) {
    AsyncContext aCtx = req.startAsync(req, res);
    userExecutor.execute(new AsyncHandler(aCtx));
  }

}

public class AsyncHandler implements Runnable {

  private AsyncContext ctx;

  public AsyncHandler(AsyncContext ctx) {
    this.ctx = ctx;
  }

  @Override
  public void run() {
    //耗时操作
    PrintWriter pw;
    try {
      pw = ctx.getResponse().getWriter();
      pw.print("done!");
      pw.flush();
      pw.close();
    } catch (IOException e) {
      e.printStackTrace();
    }
    ctx.complete();
  }
}

我们创建一个AsyncServlet,它定义了一个userExecutor线程池专门用于处理该Servlet的所有请求的耗时的逻辑操作。这样就不会占用Tomcat内部的Executor线程池,影响到对其他Servlet的处理。这种思想有点像资源隔离,耗时的操作统一由指定的线程池处理,而不要影响其它耗时少的请求处理。

Servlet的异步的实现就很好理解了,startAsync方法其实就是创建了一个异步上下文AsyncContext对象,该对象封装了请求和响应对象。然后创建一个任务用于处理耗时逻辑,后面通过AsyncContext对象获得响应对象并对客户端响应,输出“done!”。完成后要通过complete方法告诉Tomcat内部我已经处理完,Tomcat就会请求对象和响应对象进行回收处理或关闭连接。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • 在Tomcat中部署Web项目的操作方法(必看篇)

    在Tomcat中部署Web项目的操作方法(必看篇)

    下面小编就为大家带来一篇在Tomcat中部署Web项目的操作方法(必看篇)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-06-06
  • Tomcat10配置端口号为443(使用https访问)

    Tomcat10配置端口号为443(使用https访问)

    tomcat配置好了以后默认是使用8080端口访问的,本文主要介绍了Tomcat10配置端口号为443,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-07-07
  • 详解tomcat设置默认路径致使项目url冲突解决方法

    详解tomcat设置默认路径致使项目url冲突解决方法

    这篇文章主要介绍了详解tomcat设置默认路径致使项目url冲突解决方法,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2019-01-01
  • Tomcat在linux环境中开机自启(定时重启)的方法

    Tomcat在linux环境中开机自启(定时重启)的方法

    我们经常会遇到服务器断电或异常,而异常后tomcat中部署的web项目需要我手动去启动,为此,特别贡献出Linux环境中Tomcat开机自启的方式供学习使用,需要的朋友可以参考下
    2023-10-10
  • intelli idea中配置Tomcat找不到的解决办法

    intelli idea中配置Tomcat找不到的解决办法

    这篇文章主要介绍了intelli idea中配置Tomcat找不到的解决办法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-09-09
  • idea搭建项目找不到Tomcat的解决方法(图文教程)

    idea搭建项目找不到Tomcat的解决方法(图文教程)

    今天在公司中导入公司项目,需要配置tomcat,结果发现找不到,所以这篇文章主要给大家介绍了关于idea搭建项目找不到Tomcat的解决方法,需要的朋友可以参考下
    2023-07-07
  • Tomcat体系结构与插件配置教程图解

    Tomcat体系结构与插件配置教程图解

    这篇文章主要介绍了Tomcat体系结构与插件配置教程图解,包括发布程序的过程和tomcat服务器体系结构,需要的朋友可以参考下
    2018-05-05
  • 详解Tomcat是如何实现异步Servlet的

    详解Tomcat是如何实现异步Servlet的

    这篇文章主要介绍了详解Tomcat是如何实现异步Servlet的,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-10-10
  • Tomcat核心组件及应用架构详解

    Tomcat核心组件及应用架构详解

    众所周知Tomcat 就是一个 Servlet 容器,为了方便使用,他们具有http服务器的功能,所以Tomcat 就是一个“HTTP 服务器 + Servlet 容器”,我们也叫它们 Web 容器,今天重点给大家普及Tomcat核心组件及应用架构相关知识,感兴趣的朋友一起了解下吧
    2021-05-05
  • 为什么我的tomcat启动不起来

    为什么我的tomcat启动不起来

    tomcat安装好了,准备部署Web项目了,启动tomcat,发现没有反应,本文就来解决一下这个问题,感兴趣的小伙伴们可以参考一下
    2021-07-07

最新评论