伍佰目录 短网址
  当前位置:海洋目录网 » 站长资讯 » 站长资讯 » 文章详细 订阅RssFeed

Tomcat的异步Servlet实现原理

来源:本站原创 浏览:204次 时间:2021-06-12

了解Servlet规范的朋友应该都知道,从3.0开始,Tomcat的Servlet支持异步请求,或者说是Tomcat提供了异步Servlet,从而可以将一些耗时的操作放到独立的线程中,在操作完成后再返回数据,不阻塞请求的执行和返回,甚至可以基于此实现服务器推的功能。


正好不久前,网友问了异步Servlet的实现原理的问题,


我把这块梳理了一下,一起来深入了解下。


首先,先从使用异步Servlet的方式来说。

我们的使用步骤大致是:

  1. 声明Servlet,注意增加asyncSupported的属性,开启异步处理支持。@WebServlet(urlPatterns = "/demo",asyncSupported = true)

  2. 在Servlet内部,需要独立线程处理的地方,使用request获取异步Context。AsyncContext ctx = req.startAsync();

  3. 在独立线程中,使用异步Context,可以获取到其绑定的request和response,此时,就可以按照原来Servlet的写法,继续编写逻辑了。

  4. 独立线程内的操作处理完成后,需要调用异步Context的complet方法,来结束该异步线程。


    需要注意的是,异步Servlet有对应的超时时间,如果在指定的时间内没有执行完操作,response依然会走原来Servlet的结束逻辑,后续的异步操作执行完再写回的时候,可能会遇到异常。



上面是使用异步Servlet的几个主要步骤,那这个背后,是如何实现的呢?下面我们来看原理。

我们在通过Request来获取异步的Context,这一步背后发生了什么呢?

public AsyncContext startAsync() {

    ���ճ־�,�շ�ʱ��    returnstartAsync(getRequest(),response.getResponse());

    }

public AsyncContext startAsync(ServletRequest request,

            ServletResponse response) {

        if (!isAsyncSupported()) { //注意1

            throw new IllegalStateException(sm.getString("request.asyncNotSupported"));

        }

        if (asyncContext == null) {

            asyncContext = newAsyncContextImpl(this);

        }

        asyncContext.setStarted(getContext(), request, response,

                request==getRequest() && response==getResponse().getResponse());

        asyncContext.setTimeout(getConnector().getAsyncTimeout());//注意2

        return asyncContext;

    }


我们上面的注意1,就是前面步骤里提到的,配置注解的时候,要开启异步处理支持,否则在这一步直接就会抛异常。

注意2,就是前面提到的异步超时设置。


我们看到整个方法是通过当前的request和response,生成了一个对应的AsyncContext,并进行相应的配置。

在setStart方法中,执行的主要逻辑有以下几点:


public void setStarted(Context context, ServletRequest request,

            ServletResponse response, boolean originalRequestResponse) {


        this.request.getCoyoteRequest().action(

                ActionCode.ASYNC_START, this); //注意3

        List<AsyncListenerWrapper> listenersCopy = new ArrayList<>();

        listenersCopy.addAll(listeners);

        listeners.clear();

        for (AsyncListenerWrapper listener : listenersCopy) {

            try {

                listener.fireOnStartAsync(event); // 注意4

            } catch (Throwable t) {

            }

        }

    }


上面注意3的地方,在Tomcat内部,许多的响应状态通知,都是通过类似的方式,以ActionCode的方式返回的,包括前面说的complete的方法调用等。

注意4,是关于异步的Context,可以添加许多的异步Listener,在特定的事件发生时,通知到Listener。


在Servlet规范中,对startAsync方法做了这样的描述:


A call to this method ensures that the response isn't committed when the application exits out of the service method. It is committed when AsyncContext.complete is called on the returned AsyncContext or the AsyncContext times out and there are no listeners associated to handle the time out. The timer for async time outs will not start until the request and it’s associated response have returned from the container. The AsyncContext could be used to write to the response from the async thread. It can also be used to just notify that the response is not closed and committed.


以上说明,可以帮助我们理解一部分异步Servlet的实现原理,我们从源码中来梳理具体的逻辑。


我们前面的文章分析过,整个请求的处理,到Container内各组件的调用,是从EndPoint到CoyoteAdaptor。其中,在Endpoint的代码中,调用是从这一行开始:

getAdapter().service(request, response);

而在此之后,整个service执行完成时,Adaptor中,会根据请求类型判断,对于非异步和comet的请求,会进行request和response的finished的操作,同时会进行recycle的操作。

AsyncContextImpl asyncConImpl = (AsyncContextImpl)request.getAsyncContext();

            if (asyncConImpl != null) {

                async = true;

            } else if (!comet) {

                request.finishRequest();

                response.finishResponse();}

                

注意,在finishResponse的时候,outputBuffer.close();就会执行,recycle的时候,也是根据请求类型进行限制。注意CoyoteAdaptor中,会判断具体的请求种类,是否为comet或者async,

   if (!comet && !async || error.get()) {

            request.recycle();

           response.recycle();

此时,OutputBuffer就会被重置,所对于普通Servlet,以后面的其它操作就不能被继续写回了。

此外,Processor会判断请求的类型,从而决定是否进行特定的操作,比如

if (!isAsync() && !comet) {

        endRequest();

}

我们看到,在非异步请求并且也不是comet请求时,会执行endRequest操作。而具体返回Socket状态的时候,也是根据请求类型进行判断,对于异步的请求,返回的Socket状态为LONG,

else if (isAsync() || comet) {

            return SocketState.LONG;


Protocol类中,判断类型为LONG时,会进行如下操作:

if (state == SocketState.LONG) {

           // In the middle of processing a request/response. Keep the

           // socket associated with the processor. Exact requirements

           // depend on type of long poll

           connections.put(socket, processor);

           longPoll(wrapper, processor);// 配置processor的属性


从而请求可以使用原处理器进行处理。

至于异步Context执行完成后的complete操作,主要是返回一个complete的ActionCode,在Processor中据此判断请求执行完成,设置SocketStatus为OPEN_READ,开始下一轮请求的接收。


public void complete() {

        check();

        request.getCoyoteRequest().action(ActionCode.ASYNC_COMPLETE, null);

}

case ASYNC_COMPLETE: {

            socketWrapper.clearDispatches();

            if (asyncStateMachine.asyncComplete()) {

                endpoint.processSocket(this.socketWrapper, SocketStatus.OPEN_READ, true);

            }

            break;

        }


以上,即为异步Servlet的基本实现原理。总结起来主要有:

  • 主线请求执行完后Socket的longPool

  • response的状态未设置为finished,此时OutputBuffer未close,依然可以写回。


  推荐站点

  • At-lib分类目录At-lib分类目录

    At-lib网站分类目录汇集全国所有高质量网站,是中国权威的中文网站分类目录,给站长提供免费网址目录提交收录和推荐最新最全的优秀网站大全是名站导航之家

    www.at-lib.cn
  • 中国链接目录中国链接目录

    中国链接目录简称链接目录,是收录优秀网站和淘宝网店的网站分类目录,为您提供优质的网址导航服务,也是网店进行收录推广,站长免费推广网站、加快百度收录、增加友情链接和网站外链的平台。

    www.cnlink.org
  • 35目录网35目录网

    35目录免费收录各类优秀网站,全力打造互动式网站目录,提供网站分类目录检索,关键字搜索功能。欢迎您向35目录推荐、提交优秀网站。

    www.35mulu.com
  • 就要爱网站目录就要爱网站目录

    就要爱网站目录,按主题和类别列出网站。所有提交的网站都经过人工审查,确保质量和无垃圾邮件的结果。

    www.912219.com
  • 伍佰目录伍佰目录

    伍佰网站目录免费收录各类优秀网站,全力打造互动式网站目录,提供网站分类目录检索,关键字搜索功能。欢迎您向伍佰目录推荐、提交优秀网站。

    www.wbwb.net