1、什么是HTTP協(xié)議
超文本傳輸協(xié)議(HTTP)是一種通信協(xié)議敛苇,它允許將超文本標(biāo)記語言(HTML)文檔從Web服務(wù)器傳送到客戶端的瀏覽器妆绞。
2、Web服務(wù)器枫攀,瀏覽器括饶,代理服務(wù)器
當(dāng)我們打開瀏覽器,在地址欄中輸入U(xiǎn)RL来涨,然后我們就看到了網(wǎng)頁图焰。 原理是怎樣的呢?
實(shí)際上我們輸入U(xiǎn)RL后蹦掐,我們的瀏覽器給Web服務(wù)器發(fā)送了一個Request, Web服務(wù)器接到Request后進(jìn)行處理技羔,生成相應(yīng)的Response,然后發(fā)送給瀏覽器卧抗, 瀏覽器解析Response中的HTML,這樣我們就看到了網(wǎng)頁藤滥,過程如下圖所示
我們的Request 有可能是經(jīng)過了代理服務(wù)器,最后才到達(dá)Web服務(wù)器的社裆。
代理服務(wù)器就是網(wǎng)絡(luò)信息的中轉(zhuǎn)站拙绊,有什么功能呢?
1. 提高訪問速度, 大多數(shù)的代理服務(wù)器都有緩存功能标沪。
2. 突破限制榄攀, 也就是翻墻了
3. 隱藏身份。
3金句、URL詳解(地址定位符)
URL(Uniform Resource Locator) 地址用于描述一個網(wǎng)絡(luò)上的資源檩赢, 基本格式如下
schema://host[:port#]/path/.../[;url-params][?query-string][#anchor]
scheme 指定低層使用的協(xié)議(例如:http, https, ftp)
host HTTP服務(wù)器的IP地址或者域名
port# HTTP服務(wù)器的默認(rèn)端口是80,這種情況下端口號可以省略违寞。如果使用了別的端口贞瞒,必須指明,例如 http://www.cnblogs.com:8080/
path 訪問資源的路徑
url-params
query-string 發(fā)送給http服務(wù)器的數(shù)據(jù)
anchor- 錨
URL 的一個例子
http://www.mywebsite.com/sj/test;id=8079?name=sviergn&x=true#stuffSchema: http
host: www.mywebsite.com
path: /sj/test
URL params: id=8079
Query String: name=sviergn&x=true
Anchor: stuff
4趁曼、HTTP請求報(bào)文和HTTP響應(yīng)報(bào)文
HTTP報(bào)文是面向文本的憔狞,報(bào)文中的每一個字段都是一些ASCII碼串,各個字段的長度是不確定的彰阴。HTTP有兩類報(bào)文:請求報(bào)文和響應(yīng)報(bào)文。
4.1HTTP請求報(bào)文
HTTP請求報(bào)文由3部分組成(請求行+請求頭+請求體):
下面是一個實(shí)際的請求報(bào)文:
① 是請求方法拍冠,GET和POST是最常見的HTTP方法尿这,除此以外還包括DELETE、HEAD庆杜、OPTIONS射众、PUT、TRACE晃财。不過叨橱,當(dāng)前的大多數(shù)瀏覽器只支持GET和POST,Spring 3.0提供了一個HiddenHttpMethodFilter断盛,允許你通過“_method”的表單參數(shù)指定這些特殊的HTTP方法(實(shí)際上還是通過POST提交表單)罗洗。服務(wù)端配置了HiddenHttpMethodFilter后,Spring會根據(jù)_method參數(shù)指定的值模擬出相應(yīng)的HTTP方法钢猛,這樣伙菜,就可以使用這些HTTP方法對處理方法進(jìn)行映射了。
② 為請求對應(yīng)的URL地址命迈,它和報(bào)文頭的Host屬性組成完整的請求URL贩绕,③是協(xié)議名稱及版本號。
④ 是HTTP的報(bào)文頭壶愤,報(bào)文頭包含若干個屬性淑倾,格式為“屬性名:屬性值”,服務(wù)端據(jù)此獲取客戶端的信息征椒。
⑤ 是報(bào)文體娇哆,它將一個頁面表單中的組件值通過param1=value1¶m2=value2的鍵值對形式編碼成一個格式化串,它承載多個請求參數(shù)的數(shù)據(jù)。不但報(bào)文體可以傳遞請求參數(shù)迂尝,請求URL也可以通過類似于“/chapter15/user.html? param1=value1¶m2=value2”的方式傳遞請求參數(shù)脱茉。
對照上面的請求報(bào)文,我們把它進(jìn)一步分解垄开,你可以看到一幅更詳細(xì)的結(jié)構(gòu)圖:
常見的HTTP請求報(bào)文頭屬性
- Accept
請求報(bào)文可通過一個 Accept
報(bào)文頭屬性告訴服務(wù)端 客戶端接受什么類型的響應(yīng)琴许。
如下報(bào)文頭相當(dāng)于告訴服務(wù)端,俺客戶端能夠接受的響應(yīng)類型僅為純文本數(shù)據(jù)啊溉躲,你丫別發(fā)其它什么圖片啊榜田,視頻啊過來,那樣我會歇菜的~~~:
Accept:text/plain
Accept屬性的值可以為一個或多個MIME類型的值锻梳,關(guān)于MIME類型箭券,大家請參考:http://en.wikipedia.org/wiki/MIME_type
- Cookie
這是第一個要說的,客戶端的Cookie就是通過這個報(bào)文頭屬性傳給服務(wù)端的哦疑枯!如下所示:
Cookie:skin=blue; jsessionid=5F4771183629C9834F8382E23BE13C4C
注意到后臺的那個 jsessionid=5F4771183629C9834F8382E23BE13C4C
沒有辩块,服務(wù)端是怎么知道客戶端的多個請求是屬于一個Session的,原來就是通過HTTP請求報(bào)文頭的Cookie屬性的jsessionid的值關(guān)聯(lián)起來的>S馈(當(dāng)然也可以通過重寫URL的方式將會話ID附帶在每個URL的后后面哦)废亭。
- Referer
表示這個請求是從哪個URL過來的,假如你通過google搜索出一個商家的廣告頁面具钥,你對這個廣告頁面感興趣豆村,鼠標(biāo)一點(diǎn)發(fā)送一個請求報(bào)文到商家的網(wǎng)站,這個請求報(bào)文的Referer報(bào)文頭屬性值就是 http://www.google.com骂删。
很多貌似神奇的網(wǎng)頁監(jiān)控軟件(如著名的 我要啦)掌动,只要在你的網(wǎng)頁上放上一段JavaScript,就可以幫你監(jiān)控流量宁玫,全國訪問客戶的分布情況等報(bào)表和圖表粗恢,其原理就是通過這個Referer及其它一些HTTP報(bào)文頭工作的。 - Cache-Control
對緩存進(jìn)行控制撬统,如一個請求希望響應(yīng)返回的內(nèi)容在客戶端要被緩存一年适滓,或不希望被緩存就可以通過這個報(bào)文頭達(dá)到目的。
如以下設(shè)置恋追,相當(dāng)于讓服務(wù)端將對應(yīng)請求返回的響應(yīng)內(nèi)容不要在客戶端緩存(當(dāng)然響應(yīng)報(bào)文也是通過響應(yīng)報(bào)文頭通知瀏覽器客戶端的凭迹,這個下面再說):
Cache-Control: no-cache
其它報(bào)文頭屬性
參見:http://en.wikipedia.org/wiki/List_of_HTTP_header_fields
如何訪問請求報(bào)文頭
由于請求報(bào)文頭是客戶端發(fā)過來的,服務(wù)端當(dāng)然只能讀取了苦囱,以下是 HttpServletRequest
一些用于讀取請求報(bào)文頭的API:
//獲取請求報(bào)文中的屬性名稱java.util.Enumeration<java.lang.String> getHeaderNames();//獲取指定名稱的報(bào)文頭屬性的值java.lang.String getHeader(java.lang.String name)
由于一些請求報(bào)文頭屬性“太著名”了嗅绸,因此HttpServletRequest為它們提供了VIP的API:
//獲取報(bào)文頭中的Cookie(讀取Cookie的報(bào)文頭屬性) Cookie[] getCookies() ;//獲取客戶端本地化信息(讀取 Accept-Language 的報(bào)文頭屬性)java.util.Locale getLocale() //獲取請求報(bào)文體的長度(讀取Content-Length的報(bào)文頭屬性)int getContentLength();
HttpServletRequest可以通過 HttpSession getSession()
獲取請求所關(guān)聯(lián)的HttpSession,其內(nèi)部的機(jī)理是通過讀取請求報(bào)文頭中Cookie屬性的JSESSIONID的值,在服務(wù)端的一個會話Map中撕彤,根據(jù)這個JSESSIONID獲取對應(yīng)的HttpSession的對象鱼鸠。(這樣猛拴,你就不會覺得HttpSession很神秘了吧,你自己也可以做一個類似的會話管理)
4.2 HTTP響應(yīng)報(bào)文
響應(yīng)報(bào)文結(jié)構(gòu)
HTTP的響應(yīng)報(bào)文也由三部分組成(響應(yīng)行+響應(yīng)頭+響應(yīng)體):
以下是一個實(shí)際的HTTP響應(yīng)報(bào)文:
① 報(bào)文協(xié)議及版本蚀狰;
② 狀態(tài)碼及狀態(tài)描述愉昆;
③ 響應(yīng)報(bào)文頭,也是由多個屬性組成麻蹋;
④ 響應(yīng)報(bào)文體跛溉,即我們真正要的“干貨”。
響應(yīng)狀態(tài)碼
和請求報(bào)文相比扮授,響應(yīng)報(bào)文多了一個“響應(yīng)狀態(tài)碼”芳室,它以“清晰明確”的語言告訴客戶端本次請求的處理結(jié)果。
HTTP的響應(yīng)狀態(tài)碼由5段組成:
- 1xx 消息刹勃,一般是告訴客戶端堪侯,請求已經(jīng)收到了,正在處理荔仁,別急...
- 2xx 處理成功伍宦,一般表示:請求收悉、我明白你要的乏梁、請求已受理雹拄、已經(jīng)處理完成等信息.
- 3xx 重定向到其它地方。它讓客戶端再發(fā)起一個請求以完成整個處理掌呜。
- 4xx 處理發(fā)生錯誤,責(zé)任在客戶端坪哄,如客戶端的請求一個不存在的資源质蕉,客戶端未被授權(quán),禁止訪問等翩肌。
- 5xx 處理發(fā)生模暗,責(zé)任在服務(wù)端,如服務(wù)端拋出異常念祭,路由出錯兑宇,HTTP版本不支持等。
以下是幾個常見的狀態(tài)碼:
200 OK
你最希望看到的粱坤,即處理成功隶糕!303 See Other
我把你redirect到其它的頁面,目標(biāo)的URL通過響應(yīng)報(bào)文頭的Location告訴你站玄。
悟空:師傅給個桃吧枚驻,走了一天了 :relieved:唐僧:我哪有桃啊株旷!去王母娘娘那找吧 :unamused:304 Not Modified
告訴客戶端再登,你請求的這個資源至你上次取得后,并沒有更改,你直接用你本地的緩存吧锉矢,我很忙哦梯嗽,你能不能少來煩我啊沽损!404 Not Found
你最不希望看到的灯节,即找不到頁面。如你在google上找到一個頁面缠俺,點(diǎn)擊這個鏈接返回404显晶,表示這個頁面已經(jīng)被網(wǎng)站刪除了,google那邊的記錄只是美好的回憶壹士。500 Internal Server Error
看到這個錯誤磷雇,你就應(yīng)該查查服務(wù)端的日志了,肯定拋出了一堆異常躏救,別睡了唯笙,起來改BUG去吧!
有些響應(yīng)碼盒使,Web應(yīng)用服務(wù)器會自動給生成崩掘。你可以通過HttpServletResponse的API設(shè)置狀態(tài)碼:
//設(shè)置狀態(tài)碼,狀態(tài)碼在HttpServletResponse中通過一系列的常量預(yù)定義了少办,如SC_ACCEPTED苞慢,SC_OKvoid setStatus(int sc)
常見的HTTP響應(yīng)報(bào)文頭屬性
Cache-Control
響應(yīng)輸出到客戶端后,服務(wù)端通過該報(bào)文頭屬告訴客戶端如何控制響應(yīng)內(nèi)容的緩存英妓。
下面挽放,的設(shè)置讓客戶端對響應(yīng)內(nèi)容緩存3600秒,也即在3600秒內(nèi)蔓纠,如果客戶再次訪問該資源辑畦,直接從客戶端的緩存中返回內(nèi)容給客戶,不要再從服務(wù)端獲韧纫小(當(dāng)然纯出,這個功能是靠客戶端實(shí)現(xiàn)的,服務(wù)端只是通過這個屬性提示客戶端“應(yīng)該這么做”敷燎,做不做暂筝,還是決定于客戶端,如果是自己宣稱支持HTTP的客戶端硬贯,則就應(yīng)該這樣實(shí)現(xiàn))乖杠。
Cache-Control: max-age=3600ETag
一個代表響應(yīng)服務(wù)端資源(如頁面)版本的報(bào)文頭屬性,如果某個服務(wù)端資源發(fā)生變化了澄成,這個ETag就會相應(yīng)發(fā)生變化胧洒。它是Cache-Control的有益補(bǔ)充畏吓,可以讓客戶端“更智能”地處理什么時(shí)候要從服務(wù)端取資源,什么時(shí)候可以直接從緩存中返回響應(yīng)卫漫。關(guān)于ETag的說明菲饼,你可以參見:http://en.wikipedia.org/wiki/HTTP_ETag。Spring 3.0還專門為此提供了一個 org.springframework.web.filter.ShallowEtagHeaderFilter
(實(shí)現(xiàn)原理很簡單列赎,對JSP輸出的內(nèi)容MD5宏悦,這樣內(nèi)容有變化ETag就相應(yīng)變化了),用于生成響應(yīng)的ETag包吝,因?yàn)檫@東東確實(shí)可以幫助減少請求和響應(yīng)的交互饼煞。
下面是一個ETag:
ETag: "737060cd8c284d8af7ad3082f209582d"Location
我們在JSP中讓頁面Redirect到一個某個A頁面中,其實(shí)是讓客戶端再發(fā)一個請求到A頁面诗越,這個需要Redirect的A頁面的URL砖瞧,其實(shí)就是通過響應(yīng)報(bào)文頭的Location屬性告知客戶端的,如下的報(bào)文頭屬性嚷狞,將使客戶端redirect到iteye的首頁中:
Location: https://github.com/biezhi/jbSet-Cookie
服務(wù)端可以設(shè)置客戶端的Cookie块促,其原理就是通過這個響應(yīng)報(bào)文頭屬性實(shí)現(xiàn)的:
Set-Cookie: UserID=Jack; Max-Age=3600; Version=1
其它HTTP請求報(bào)文頭屬性
更多其它的HTTP請求頭報(bào)文,參見:http://en.wikipedia.org/wiki/List_of_HTTP_header_fields
如何寫HTTP請求報(bào)文頭
在服務(wù)端可以通過HttpServletResponse的API寫響應(yīng)報(bào)文頭的屬性:
//添加一個響應(yīng)報(bào)文頭屬性void setHeader(String name, String value)
像Cookie床未,Location這些響應(yīng)頭是有福之人竭翠,HttpServletResponse為它們都提供了VIP(非API 哈):
//添加Cookie報(bào)文頭屬性void addCookie(Cookie cookie) //不但會設(shè)置Location的響應(yīng)報(bào)文頭,還會生成303的狀態(tài)碼呢薇搁,兩者天仙配呢void sendRedirect(String location)
摘自:http://www.cnblogs.com/ImBit/p/5513401.html#two.one
http://blog.csdn.net/zhangliang_571/article/details/23508953
https://yq.aliyun.com/articles/44672