HTTP請(qǐng)求行呀邢、請(qǐng)求頭豹绪、請(qǐng)求體詳解

HTTP請(qǐng)求報(bào)文解剖

HTTP請(qǐng)求報(bào)文由3部分組成(請(qǐng)求行+請(qǐng)求頭+請(qǐng)求體):


請(qǐng)求報(bào)文.jpg

下面是一個(gè)實(shí)際的請(qǐng)求報(bào)文:


實(shí)際的請(qǐng)求報(bào)文.jpg

①是請(qǐng)求方法蝉衣,GET和POST是最常見的HTTP方法巷蚪,除此以外還包括DELETE、HEAD剪验、OPTIONS功戚、PUT似嗤、TRACE。不過(guò)乘粒,當(dāng)前的大多數(shù)瀏覽器只支持GET和POST灯萍,Spring 3.0提供了一個(gè)HiddenHttpMethodFilter每聪,允許你通過(guò)“_method”的表單參數(shù)指定這些特殊的HTTP方法(實(shí)際上還是通過(guò)POST提交表單)。服務(wù)端配置了HiddenHttpMethodFilter后绑洛,Spring會(huì)根據(jù)_method參數(shù)指定的值模擬出相應(yīng)的HTTP方法真屯,這樣穷娱,就可以使用這些HTTP方法對(duì)處理方法進(jìn)行映射了运沦。

②為請(qǐng)求對(duì)應(yīng)的URL地址茶袒,它和報(bào)文頭的Host屬性組成完整的請(qǐng)求URL,③是協(xié)議名稱及版本號(hào)亡资。

④是HTTP的報(bào)文頭,報(bào)文頭包含若干個(gè)屬性嗦董,格式為“屬性名:屬性值”京革,服務(wù)端據(jù)此獲取客戶端的信息幸斥。

⑤是報(bào)文體,它將一個(gè)頁(yè)面表單中的組件值通過(guò)param1=value1&param2=value2的鍵值對(duì)形式編碼成一個(gè)格式化串廊勃,它承載多個(gè)請(qǐng)求參數(shù)的數(shù)據(jù)坡垫。不但報(bào)文體可以傳遞請(qǐng)求參數(shù)画侣,請(qǐng)求URL也可以通過(guò)類似于“/chapter15/user.html? param1=value1&param2=value2”的方式傳遞請(qǐng)求參數(shù)配乱。

對(duì)照上面的請(qǐng)求報(bào)文,我們把它進(jìn)一步分解的诵,你可以看到一幅更詳細(xì)的結(jié)構(gòu)圖:

結(jié)構(gòu)圖.jpg

HttpWatch是強(qiáng)大的網(wǎng)頁(yè)數(shù)據(jù)分析工具西疤,安裝后將集成到Internet Explorer工具欄中休溶。它不用代理服務(wù)器或一些復(fù)雜的網(wǎng)絡(luò)監(jiān)控工具扰她,就能抓取請(qǐng)求及響應(yīng)的完整信息徒役,包括Cookies忧勿、消息頭瞻讽、查詢參數(shù)、響應(yīng)報(bào)文等晌砾,是Web應(yīng)用開發(fā)人員的必備工具养匈。

HTTP請(qǐng)求報(bào)文頭屬性

報(bào)文頭屬性是什么東西呢都伪?我們不妨以一個(gè)小故事來(lái)說(shuō)明吧。

引用
快到中午了楣嘁,張三豐不想去食堂吃飯逐虚,于是打電話叫外賣:老板谆膳,我要一份[魚香肉絲],要12:30之前給我送過(guò)來(lái)哦买雾,我在江湖湖公司研發(fā)部杨帽,叫張三豐注盈。

這里,你要[魚香肉絲]相當(dāng)于HTTP報(bào)文體僚饭,而“12:30之前送過(guò)來(lái)”,你叫“張三豐”等信息就相當(dāng)于HTTP的報(bào)文頭苇瓣。它們是一些附屬信息击罪,幫忙你和飯店老板順利完成這次交易贪薪。

請(qǐng)求HTTP報(bào)文和響應(yīng)HTTP報(bào)文都擁有若干個(gè)報(bào)文關(guān)屬性,它們是為協(xié)助客戶端及服務(wù)端交易的一些附屬信息。

常見的HTTP請(qǐng)求報(bào)文頭屬性

Accept

請(qǐng)求報(bào)文可通過(guò)一個(gè)“Accept”報(bào)文頭屬性告訴服務(wù)端 客戶端接受什么類型的響應(yīng)槽唾。

如下報(bào)文頭相當(dāng)于告訴服務(wù)端光涂,俺客戶端能夠接受的響應(yīng)類型僅為純文本數(shù)據(jù)啊,你丫別發(fā)其它什么圖片啊钝计,視頻啊過(guò)來(lái)私恬,那樣我會(huì)歇菜的~~~:

Accept:text/plain  

Accept屬性的值可以為一個(gè)或多個(gè)MIME類型的值炼吴,關(guān)于MIME類型,大家請(qǐng)參考:http://en.wikipedia.org/wiki/MIME_type

Cookie
客戶端的Cookie就是通過(guò)這個(gè)報(bào)文頭屬性傳給服務(wù)端的哦荣德!如下所示:

Cookie: $Version=1; Skin=new;jsessionid=5F4771183629C9834F8382E23BE13C4C  

服務(wù)端是怎么知道客戶端的多個(gè)請(qǐng)求是隸屬于一個(gè)Session呢涮瞻?注意到后臺(tái)的那個(gè)jsessionid=5F4771183629C9834F8382E23BE13C4C木有假褪?原來(lái)就是通過(guò)HTTP請(qǐng)求報(bào)文頭的Cookie屬性的jsessionid的值關(guān)聯(lián)起來(lái)的!(當(dāng)然也可以通過(guò)重寫URL的方式將會(huì)話ID附帶在每個(gè)URL的后面哦)艇抠。

Referer
表示這個(gè)請(qǐng)求是從哪個(gè)URL過(guò)來(lái)的,假如你通過(guò)google搜索出一個(gè)商家的廣告頁(yè)面异剥,你對(duì)這個(gè)廣告頁(yè)面感興趣絮重,鼠標(biāo)一點(diǎn)發(fā)送一個(gè)請(qǐng)求報(bào)文到商家的網(wǎng)站青伤,這個(gè)請(qǐng)求報(bào)文的Referer報(bào)文頭屬性值就是http://www.google.com
引用

唐僧到了西天.
如來(lái)問(wèn):儂是不是從東土大唐來(lái)昂鸥堋丰歌?
唐僧:厲害立帖!你咋知道的!
如來(lái):呵呵堂飞,我偷看了你的Referer...
很多貌似神奇的網(wǎng)頁(yè)監(jiān)控軟件(如著名的 我要啦)绑咱,只要在你的網(wǎng)頁(yè)上放上一段JavaScript羡玛,就可以幫你監(jiān)控流量,全國(guó)訪問(wèn)客戶的分布情況等報(bào)表和圖表薄榛,其原理就是通過(guò)這個(gè)Referer及其它一些HTTP報(bào)文頭工作的敞恋。

Cache-Control

對(duì)緩存進(jìn)行控制谋右,如一個(gè)請(qǐng)求希望響應(yīng)返回的內(nèi)容在客戶端要被緩存一年,或不希望被緩存就可以通過(guò)這個(gè)報(bào)文頭達(dá)到目的啸蜜。

如以下設(shè)置衬横,相當(dāng)于讓服務(wù)端將對(duì)應(yīng)請(qǐng)求返回的響應(yīng)內(nèi)容不要在客戶端緩存:

Cache-Control: no-cache  

其它請(qǐng)求報(bào)文頭屬性

參見:http://en.wikipedia.org/wiki/List_of_HTTP_header_fields

如何訪問(wèn)請(qǐng)求報(bào)文頭

由于請(qǐng)求報(bào)文頭是客戶端發(fā)過(guò)來(lái)的,服務(wù)端當(dāng)然只能讀取了遥诉,以下是HttpServletRequest一些用于讀取請(qǐng)求報(bào)文頭的API:

     //獲取請(qǐng)求報(bào)文中的屬性名稱  
    java.util.Enumeration<java.lang.String>   getHeaderNames();  
      
    //獲取指定名稱的報(bào)文頭屬性的值  
    java.lang.String getHeader(java.lang.String name)   

由于一些請(qǐng)求報(bào)文頭屬性“太著名”了矮锈,因此HttpServletRequest為它們提供了VIP的API:

 //獲取報(bào)文頭中的Cookie(讀取Cookie的報(bào)文頭屬性)  
 Cookie[]   getCookies() ;  
  
//獲取客戶端本地化信息(讀取 Accept-Language 的報(bào)文頭屬性)  
java.util.Locale    getLocale()   
  
//獲取請(qǐng)求報(bào)文體的長(zhǎng)度(讀取Content-Length的報(bào)文頭屬性)  
int getContentLength(); 

HttpServletRequest可以通過(guò)

HttpSession getSession()  

獲取請(qǐng)求所關(guān)聯(lián)的HttpSession,其內(nèi)部的機(jī)理是通過(guò)讀取請(qǐng)求報(bào)文頭中Cookie屬性的JSESSIONID的值苞笨,在服務(wù)端的一個(gè)會(huì)話Map中子眶,根據(jù)這個(gè)JSESSIONID獲取對(duì)應(yīng)的HttpSession的對(duì)象。(這樣,你就不會(huì)覺得HttpSession很神秘了吧芝加,你自己也可以做一個(gè)類似的會(huì)話管理 ! )

HTTP響應(yīng)報(bào)文解剖

響應(yīng)報(bào)文結(jié)構(gòu)

HTTP的響應(yīng)報(bào)文也由三部分組成(響應(yīng)行+響應(yīng)頭+響應(yīng)體):

響應(yīng)報(bào)文結(jié)構(gòu).jpg

以下是一個(gè)實(shí)際的HTTP響應(yīng)報(bào)文:
相應(yīng)報(bào)文.jpg

①報(bào)文協(xié)議及版本将塑;
②狀態(tài)碼及狀態(tài)描述蝌麸;
③響應(yīng)報(bào)文頭,也是由多個(gè)屬性組成敢辩;
④響應(yīng)報(bào)文體戚长,即我們真正要的“干貨”怠苔。

響應(yīng)狀態(tài)碼

和請(qǐng)求報(bào)文相比,響應(yīng)報(bào)文多了一個(gè)“響應(yīng)狀態(tài)碼”迫肖,它以“清晰明確”的語(yǔ)言告訴客戶端本次請(qǐng)求的處理結(jié)果蟆湖。

HTTP的響應(yīng)狀態(tài)碼由5段組成:

  • 1xx 消息,一般是告訴客戶端稠集,請(qǐng)求已經(jīng)收到了饥瓷,正在處理,別急...
  • 2xx 處理成功晦鞋,一般表示:請(qǐng)求收悉棺克、我明白你要的、請(qǐng)求已受理娜谊、已經(jīng)處理完成等信息.
  • 3xx 重定向到其它地方。它讓客戶端再發(fā)起一個(gè)請(qǐng)求以完成整個(gè)處理湾趾。
  • 4xx 處理發(fā)生錯(cuò)誤派草,責(zé)任在客戶端,如客戶端的請(qǐng)求一個(gè)不存在的資源艺普,客戶端未被授權(quán)歧譬,禁止訪問(wèn)等搏存。
  • 5xx 處理發(fā)生錯(cuò)誤,責(zé)任在服務(wù)端面氓,如服務(wù)端拋出異常,路由出錯(cuò)舌界,HTTP版本不支持等呻拌。

以下是幾個(gè)常見的狀態(tài)碼:

200 OK

你最希望看到的,即處理成功靴拱!

303 See Other

我把你redirect到其它的頁(yè)面猾普,目標(biāo)的URL通過(guò)響應(yīng)報(bào)文頭的Location告訴你。

引用

悟空:師傅給個(gè)桃吧偎窘,走了一天了!
唐僧:我哪有桃澳爸掖肋!去王母娘娘那找吧!

304 Not Modified

告訴客戶端,你請(qǐng)求的這個(gè)資源至你上次取得后浙芙,并沒有更改籽腕,你直接用你本地的緩存吧皇耗,我很忙哦揍很,你能不能少來(lái)煩我爸匣凇!

404 Not Found

你最不希望看到的阶界,即找不到頁(yè)面。如你在google上找到一個(gè)頁(yè)面膘融,點(diǎn)擊這個(gè)鏈接返回404,表示這個(gè)頁(yè)面已經(jīng)被網(wǎng)站刪除了春畔,google那邊的記錄只是美好的回憶律姨。

500 Internal Server Error

看到這個(gè)錯(cuò)誤臼疫,你就應(yīng)該查查服務(wù)端的日志了,肯定拋出了一堆異常缓淹,別睡了讯壶,起來(lái)改BUG去吧湾盗!

其它的狀態(tài)碼參見:http://en.wikipedia.org/wiki/List_of_HTTP_status_codes

有些響應(yīng)碼,Web應(yīng)用服務(wù)器會(huì)自動(dòng)給生成躏吊。你可以通過(guò)HttpServletResponse的API設(shè)置狀態(tài)碼:

 //設(shè)置狀態(tài)碼帐萎,狀態(tài)碼在HttpServletResponse中通過(guò)一系列的常量預(yù)定義了疆导,如SC_ACCEPTED,SC_OK  
void    setStatus(int sc)

常見的HTTP響應(yīng)報(bào)文頭屬性

Cache-Control

響應(yīng)輸出到客戶端后悠菜,服務(wù)端通過(guò)該報(bào)文頭屬告訴客戶端如何控制響應(yīng)內(nèi)容的緩存败富。

下面,的設(shè)置讓客戶端對(duì)響應(yīng)內(nèi)容緩存3600秒芬骄,也即在3600秒內(nèi),如果客戶再次訪問(wèn)該資源匪蟀,直接從客戶端的緩存中返回內(nèi)容給客戶宰僧,不要再?gòu)姆?wù)端獲取(當(dāng)然段化,這個(gè)功能是靠客戶端實(shí)現(xiàn)的造成,服務(wù)端只是通過(guò)這個(gè)屬性提示客戶端“應(yīng)該這么做”晒屎,做不做,還是決定于客戶端蕴轨,如果是自己宣稱支持HTTP的客戶端骇吭,則就應(yīng)該這樣實(shí)現(xiàn))。

Cache-Control: max-age=3600  

ETag

一個(gè)代表響應(yīng)服務(wù)端資源(如頁(yè)面)版本的報(bào)文頭屬性棘脐,如果某個(gè)服務(wù)端資源發(fā)生變化了蛀缝,這個(gè)ETag就會(huì)相應(yīng)發(fā)生變化目代。它是Cache-Control的有益補(bǔ)充,可以讓客戶端“更智能”地處理什么時(shí)候要從服務(wù)端取資源,什么時(shí)候可以直接從緩存中返回響應(yīng)潭苞。

關(guān)于ETag的說(shuō)明此疹,你可以參見:http://en.wikipedia.org/wiki/HTTP_ETag遮婶。
Spring 3.0還專門為此提供了一個(gè)org.springframework.web.filter.ShallowEtagHeaderFilter(實(shí)現(xiàn)原理很簡(jiǎn)單旗扑,對(duì)JSP輸出的內(nèi)容MD5慈省,這樣內(nèi)容有變化ETag就相應(yīng)變化了),用于生成響應(yīng)的ETag袱衷,因?yàn)檫@東東確實(shí)可以幫助減少請(qǐng)求和響應(yīng)的交互笑窜。

下面是一個(gè)ETag:

ETag: "737060cd8c284d8af7ad3082f209582d"

Location

我們?cè)贘SP中讓頁(yè)面Redirect到一個(gè)某個(gè)A頁(yè)面中排截,其實(shí)是讓客戶端再發(fā)一個(gè)請(qǐng)求到A頁(yè)面,這個(gè)需要Redirect到的A頁(yè)面的URL脱吱,其實(shí)就是通過(guò)響應(yīng)報(bào)文頭的Location屬性告知客戶端的艳悔,如下的報(bào)文頭屬性,將使客戶端redirect到iteye的首頁(yè)中:

Location: http://www.iteye.com  

Set-Cookie

服務(wù)端可以設(shè)置客戶端的Cookie抡锈,其原理就是通過(guò)這個(gè)響應(yīng)報(bào)文頭屬性實(shí)現(xiàn)的:

Set-Cookie: UserID=JohnDoe; Max-Age=3600; Version=1  

其它HTTP響應(yīng)報(bào)文頭屬性

更多其它的HTTP響應(yīng)頭報(bào)文乔外,參見:http://en.wikipedia.org/wiki/List_of_HTTP_header_fields

如何寫HTTP請(qǐng)求報(bào)文頭

在服務(wù)端可以通過(guò)HttpServletResponse的API寫響應(yīng)報(bào)文頭的屬性:

 //添加一個(gè)響應(yīng)報(bào)文頭屬性  
void    setHeader(String name, String value) 

像Cookie杨幼,Location這些響應(yīng)都是有福之人,HttpServletResponse為它們都提供了VIP版的API:

     //添加Cookie報(bào)文頭屬性  
    void addCookie(Cookie cookie)   
      
    //不但會(huì)設(shè)置Location的響應(yīng)報(bào)文頭四瘫,還會(huì)生成303的狀態(tài)碼呢欲逃,兩者天仙配呢  
    void    sendRedirect(String location)   
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市洗做,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌撰筷,老刑警劉巖畦徘,帶你破解...
    沈念sama閱讀 219,427評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件旧烧,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡掘剪,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,551評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門廉赔,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)蜡塌,“玉大人勿负,你說(shuō)我怎么就攤上這事±拍Γ” “怎么了锭硼?”我有些...
    開封第一講書人閱讀 165,747評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵檀头,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我搭独,道長(zhǎng)廊镜,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,939評(píng)論 1 295
  • 正文 為了忘掉前任惊奇,我火速辦了婚禮颂郎,結(jié)果婚禮上乓序,老公的妹妹穿的比我還像新娘坎背。我一直安慰自己,他們只是感情好陨献,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,955評(píng)論 6 392
  • 文/花漫 我一把揭開白布懂更。 她就那樣靜靜地躺著沮协,像睡著了一般。 火紅的嫁衣襯著肌膚如雪慷暂。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,737評(píng)論 1 305
  • 那天奸腺,我揣著相機(jī)與錄音洋机,去河邊找鬼洋魂。 笑死,一個(gè)胖子當(dāng)著我的面吹牛副砍,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播角骤,決...
    沈念sama閱讀 40,448評(píng)論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼邦尊,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了链峭?” 一聲冷哼從身側(cè)響起又沾,我...
    開封第一講書人閱讀 39,352評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤杖刷,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后役听,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體表窘,經(jīng)...
    沈念sama閱讀 45,834評(píng)論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡蚊丐,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,992評(píng)論 3 338
  • 正文 我和宋清朗相戀三年麦备,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片黍匾。...
    茶點(diǎn)故事閱讀 40,133評(píng)論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡锐涯,死狀恐怖填物,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情升薯,我是刑警寧澤击困,帶...
    沈念sama閱讀 35,815評(píng)論 5 346
  • 正文 年R本政府宣布,位于F島的核電站谅海,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏胁赢。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,477評(píng)論 3 331
  • 文/蒙蒙 一徒河、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧送漠,春花似錦闽寡、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,022評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)羡微。三九已至,卻和暖如春妈倔,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背毅哗。 一陣腳步聲響...
    開封第一講書人閱讀 33,147評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工黎做, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人蒸殿。 一個(gè)月前我還...
    沈念sama閱讀 48,398評(píng)論 3 373
  • 正文 我出身青樓酥艳,卻偏偏與公主長(zhǎng)得像爬骤,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子骤铃,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,077評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容