欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  网络运营

tomcat何时写回响应数据报的详析

程序员文章站 2022-03-18 14:41:57
疑问的产生 这个疑问是我在写文件下载的时候产生的,我是用httpservletresponse获取到outputstream,然后利用outputstream直接写数...

疑问的产生

这个疑问是我在写文件下载的时候产生的,我是用httpservletresponse获取到outputstream,然后利用outputstream直接写数据的。当时我就想这个outputstream是不是就是对应的socket连接的outputstream。即是不是的程序在用stream写的时候,数据也同时在发?

response的outputstream把数据写到哪去?

于是我看了下httpservletresponse的getoutputstream方法,看看它注释是怎么说的。

/**
  * returns a {@link servletoutputstream} suitable for writing binary 
  * data in the response. the servlet container does not encode the
  * binary data. 
  *
  * <p> calling flush() on the servletoutputstream commits the response.
  *
  * either this method or {@link #getwriter} may 
  * be called to write the body, not both, except when {@link #reset}
  * has been called.
  *
  * @return a {@link servletoutputstream} for writing binary data 
  *
  * @exception illegalstateexception if the <code>getwriter</code> method
  * has been called on this response
  *
  * @exception ioexception if an input or output exception occurred
  *
  * @see #getwriter
  * @see #reset
  */
 public servletoutputstream getoutputstream() throws ioexception;

以上,注释有说明是outputstream是用来写响应body内容的,也有提到flush()方法,说明肯定是有缓冲的,所以应该不是直接操作socket写数据。我猜测应该是有一个字节数组用来暂时存储,然后统一flush。但是还是不太确定,于是简单翻阅了下tomcat源码。

tomcat何时写回响应数据报的详析

找到servletoutputstream的实现类coyoteouputstream。它实现了outputstream的抽象方法write,把数据写入到outputbuffer类型的字段中存着。而这个outputbuffer对象来自于coyote/response。其实这个outputbuffer也只是一个接口,具体实现一直向下翻是streamoutputbuffer。数据大小没有限制,是用链表存储的,每个链表节点存储8196字节。

什么时候把响应数据报返回给客户端?

其实就是查看,它是何时调用outputbuffer的flush方法的。我逐层查看,最终定位到了connector/response的finishresponse()方法。这个方法,会先发送响应行和响应头。然后再发送响应body。tomcat的源码我看的不多,这里找到一张不错的时序图,描述的是一个http请求的处理过程。如下,我们把重点放在servlet的service方法调用,和response的fininshresponse方法调用上。可以得到,在service方法返回后,执行的就是finishresponse操作。也就是说,当servlet程序处理完这个请求后,tomcat就会把响应结果发回客户端

注意:servlet的程序不参与底层数据的收发,或者说不控制

tomcat何时写回响应数据报的详析

servlet的service方法调用在图中哪里?

包含在applicationfilterchain的internaldofilter方法中。

servlet程序处理请求指的是什么?

根本上servlet程序做的工作就是,根据request的信息,填充response信息而已。

servlet程序与spring mvc是什么关系?

spring mvc底层还是serlvet,它是把所有请求都用一个servlet处理,这个servlet叫做dispatcherservlet,而它又把请求分发给对应的@requestmapping标注的方法进行处理。整体上来说就是完成一个service方法的调用。

那mvc的返回页面,返回rest数据是怎么回事?

返回页面就是把页面数据写入到响应body中;@responsebody注解,实际上就是把@requestmapping标注的方法的返回值转为json字符串写入到响应body中。这里的响应body指的就是前文中的outputbuffer.

tomcat与servlet程序的职责

《how tomcat works》中讲到,servlet容器(tomcat就是一种servlet容器)的任务有概括地讲有三个

1.创建一个request对象,并填充相关信息(parameters、headers、cookie、uri等)

2.创建一个response对象

3.调用与此请求关联的servlet的service方法,把request和response传给它。

这里我用自己的话讲一下:当浏览器向服务端发来一个请求时,服务端会将请求数据报的内容解析出来,创建一个填充有请求信息的request对象,同时创建一个"空的"response对象,然后把这两个对象传给servlet的service方法,让它来完成response对象的填充,最后把response数据发送给客户端。

为什么要传request对象?

你不传request对象,servlet程序就不知道该填充什么。换句话说,它不知道你到底想要什么资源。

tomcat是如何找到请求关联的servlet的?

我们知道,tomcat在开发的时候不可能知道你会往它里面部署什么项目,servlet程序叫什么。所以它不可能硬编码来调用service方法,它所使用的就是反射机制。

想想在使用spring boot框架开发之前,我们是怎么部署项目的?就是把项目打包,然后放到tomcat的webapp目录下。跑起来后,项目对应的url就是localhost:8080/projectname/xxx这样是吧。而且,在项目中,不管是注解式的,还是web.xml式,都会配置servlet程序的映射。把url映射到某个servlet类文件。

当请求来临时,先根据projectname找到对应项目,再根据后续的url映射到对应的servlet类名。之后tomcat就会利用反射机制加载servlet类文件,获取实例,然后再调用service方法。

coyote/response、connector/response、connector/responsefacade之间的关系?

coyote/response主要就是跟底层的数据传递挂钩的,而connector/response是coyote/response的上层包装,它实现了httpservletresponse接口。但是如果将它直接传给service方法,则害怕用户直接将httpservletresponse强转为connector/response,直接调用底层的一些方法。所以引入了一个使用"facade模式",将connector/response除了httpservletresponse接口定义的public方法都屏蔽掉。也就是说,传递给service的实际上是connector/responsefacade对象,就算强转为实际类型,也只能看到httpservletresponse接口定义的方法。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对的支持。