人妖在线一区,国产日韩欧美一区二区综合在线,国产啪精品视频网站免费,欧美内射深插日本少妇

新聞動(dòng)態(tài)

tomcat何時(shí)寫回響應(yīng)數(shù)據(jù)報(bào)的詳析

發(fā)布日期:2022-03-10 13:49 | 文章來(lái)源:源碼之家

疑問(wèn)的產(chǎn)生

這個(gè)疑問(wèn)是我在寫文件下載的時(shí)候產(chǎn)生的,我是用HttpServletResponse獲取到Outputstream,然后利用OutputStream直接寫數(shù)據(jù)的。當(dāng)時(shí)我就想這個(gè)OutputStream是不是就是對(duì)應(yīng)的Socket連接的OutputStream。即是不是的程序在用stream寫的時(shí)候,數(shù)據(jù)也同時(shí)在發(fā)?

Response的OutputStream把數(shù)據(jù)寫到哪去?

于是我看了下HttpServletResponse的getOutputStream方法,看看它注釋是怎么說(shuō)的。

/**
  * 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;

以上,注釋有說(shuō)明是OutputStream是用來(lái)寫響應(yīng)body內(nèi)容的,也有提到flush()方法,說(shuō)明肯定是有緩沖的,所以應(yīng)該不是直接操作socket寫數(shù)據(jù)。我猜測(cè)應(yīng)該是有一個(gè)字節(jié)數(shù)組用來(lái)暫時(shí)存儲(chǔ),然后統(tǒng)一flush。但是還是不太確定,于是簡(jiǎn)單翻閱了下tomcat源碼。

找到ServletOutputStream的實(shí)現(xiàn)類CoyoteOuputStream。它實(shí)現(xiàn)了OutputStream的抽象方法write,把數(shù)據(jù)寫入到OutputBuffer類型的字段中存著。而這個(gè)OutputBuffer對(duì)象來(lái)自于coyote/Response。其實(shí)這個(gè)OutputBuffer也只是一個(gè)接口,具體實(shí)現(xiàn)一直向下翻是StreamOutputBuffer。數(shù)據(jù)大小沒(méi)有限制,是用鏈表存儲(chǔ)的,每個(gè)鏈表節(jié)點(diǎn)存儲(chǔ)8196字節(jié)。

什么時(shí)候把響應(yīng)數(shù)據(jù)報(bào)返回給客戶端?

其實(shí)就是查看,它是何時(shí)調(diào)用OutputBuffer的flush方法的。我逐層查看,最終定位到了connector/Response的finishResponse()方法。這個(gè)方法,會(huì)先發(fā)送響應(yīng)行和響應(yīng)頭。然后再發(fā)送響應(yīng)body。Tomcat的源碼我看的不多,這里找到一張不錯(cuò)的時(shí)序圖,描述的是一個(gè)HTTP請(qǐng)求的處理過(guò)程。如下,我們把重點(diǎn)放在servlet的service方法調(diào)用,和Response的fininshResponse方法調(diào)用上??梢缘玫?,在service方法返回后,執(zhí)行的就是finishResponse操作。也就是說(shuō),當(dāng)servlet程序處理完這個(gè)請(qǐng)求后,tomcat就會(huì)把響應(yīng)結(jié)果發(fā)回客戶端

注意:servlet的程序不參與底層數(shù)據(jù)的收發(fā),或者說(shuō)不控制

servlet的service方法調(diào)用在圖中哪里?

包含在ApplicationFilterChain的internalDoFilter方法中。

servlet程序處理請(qǐng)求指的是什么?

根本上servlet程序做的工作就是,根據(jù)Request的信息,填充Response信息而已。

servlet程序與Spring MVC是什么關(guān)系?

Spring MVC底層還是Serlvet,它是把所有請(qǐng)求都用一個(gè)servlet處理,這個(gè)servlet叫做DispatcherServlet,而它又把請(qǐng)求分發(fā)給對(duì)應(yīng)的@RequestMapping標(biāo)注的方法進(jìn)行處理。整體上來(lái)說(shuō)就是完成一個(gè)service方法的調(diào)用。

那MVC的返回頁(yè)面,返回REST數(shù)據(jù)是怎么回事?

返回頁(yè)面就是把頁(yè)面數(shù)據(jù)寫入到響應(yīng)Body中;@ResponseBody注解,實(shí)際上就是把@RequestMapping標(biāo)注的方法的返回值轉(zhuǎn)為JSON字符串寫入到響應(yīng)Body中。這里的響應(yīng)Body指的就是前文中的OutputBuffer.

Tomcat與Servlet程序的職責(zé)

《How Tomcat works》中講到,Servlet容器(Tomcat就是一種Servlet容器)的任務(wù)有概括地講有三個(gè)

1.創(chuàng)建一個(gè)Request對(duì)象,并填充相關(guān)信息(parameters、headers、cookie、uri等)

2.創(chuàng)建一個(gè)Response對(duì)象

3.調(diào)用與此請(qǐng)求關(guān)聯(lián)的Servlet的service方法,把Request和Response傳給它。

這里我用自己的話講一下:當(dāng)瀏覽器向服務(wù)端發(fā)來(lái)一個(gè)請(qǐng)求時(shí),服務(wù)端會(huì)將請(qǐng)求數(shù)據(jù)報(bào)的內(nèi)容解析出來(lái),創(chuàng)建一個(gè)填充有請(qǐng)求信息的Request對(duì)象,同時(shí)創(chuàng)建一個(gè)"空的"Response對(duì)象,然后把這兩個(gè)對(duì)象傳給servlet的service方法,讓它來(lái)完成Response對(duì)象的填充,最后把Response數(shù)據(jù)發(fā)送給客戶端。

為什么要傳Request對(duì)象?

你不傳Request對(duì)象,Servlet程序就不知道該填充什么。換句話說(shuō),它不知道你到底想要什么資源。

Tomcat是如何找到請(qǐng)求關(guān)聯(lián)的Servlet的?

我們知道,Tomcat在開(kāi)發(fā)的時(shí)候不可能知道你會(huì)往它里面部署什么項(xiàng)目,servlet程序叫什么。所以它不可能硬編碼來(lái)調(diào)用service方法,它所使用的就是反射機(jī)制。

想想在使用spring boot框架開(kāi)發(fā)之前,我們是怎么部署項(xiàng)目的?就是把項(xiàng)目打包,然后放到Tomcat的webapp目錄下。跑起來(lái)后,項(xiàng)目對(duì)應(yīng)的URL就是localhost:8080/projectName/xxx這樣是吧。而且,在項(xiàng)目中,不管是注解式的,還是web.xml式,都會(huì)配置Servlet程序的映射。把URL映射到某個(gè)Servlet類文件。

當(dāng)請(qǐng)求來(lái)臨時(shí),先根據(jù)projectName找到對(duì)應(yīng)項(xiàng)目,再根據(jù)后續(xù)的URL映射到對(duì)應(yīng)的Servlet類名。之后Tomcat就會(huì)利用反射機(jī)制加載Servlet類文件,獲取實(shí)例,然后再調(diào)用service方法。

coyote/Response、connector/Response、connector/ResponseFacade之間的關(guān)系?

coyote/Response主要就是跟底層的數(shù)據(jù)傳遞掛鉤的,而connector/Response是coyote/Response的上層包裝,它實(shí)現(xiàn)了HttpServletResponse接口。但是如果將它直接傳給service方法,則害怕用戶直接將HttpServletResponse強(qiáng)轉(zhuǎn)為connector/Response,直接調(diào)用底層的一些方法。所以引入了一個(gè)使用"Facade模式",將connector/Response除了HttpServletResponse接口定義的public方法都屏蔽掉。也就是說(shuō),傳遞給service的實(shí)際上是connector/ResponseFacade對(duì)象,就算強(qiáng)轉(zhuǎn)為實(shí)際類型,也只能看到HttpServletResponse接口定義的方法。

總結(jié)

以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)本站的支持。

版權(quán)聲明:本站文章來(lái)源標(biāo)注為YINGSOO的內(nèi)容版權(quán)均為本站所有,歡迎引用、轉(zhuǎn)載,請(qǐng)保持原文完整并注明來(lái)源及原文鏈接。禁止復(fù)制或仿造本網(wǎng)站,禁止在非www.sddonglingsh.com所屬的服務(wù)器上建立鏡像,否則將依法追究法律責(zé)任。本站部分內(nèi)容來(lái)源于網(wǎng)友推薦、互聯(lián)網(wǎng)收集整理而來(lái),僅供學(xué)習(xí)參考,不代表本站立場(chǎng),如有內(nèi)容涉嫌侵權(quán),請(qǐng)聯(lián)系alex-e#qq.com處理。

相關(guān)文章

實(shí)時(shí)開(kāi)通

自選配置、實(shí)時(shí)開(kāi)通

免備案

全球線路精選!

全天候客戶服務(wù)

7x24全年不間斷在線

專屬顧問(wèn)服務(wù)

1對(duì)1客戶咨詢顧問(wèn)

在線
客服

在線客服:7*24小時(shí)在線

客服
熱線

400-630-3752
7*24小時(shí)客服服務(wù)熱線

關(guān)注
微信

關(guān)注官方微信
頂部