服务器之家:专注于服务器技术及软件下载分享
分类导航

云服务器|WEB服务器|FTP服务器|邮件服务器|虚拟主机|服务器安全|DNS服务器|服务器知识|Nginx|IIS|Tomcat|

服务器之家 - 服务器技术 - Tomcat - tomcat何时写回响应数据报的详析

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

2021-09-06 18:07猫毛·波拿巴 Tomcat

这篇文章主要给大家介绍了关于tomcat是何时写回响应数据报的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用tomcat具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧

疑问的产生

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

response的outputstream把数据写到哪去?

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

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
/**
  * 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接口定义的方法。

总结

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

原文链接:https://www.cnblogs.com/longfurcat/p/10371283.html

延伸 · 阅读

精彩推荐