摘要:前言最近看了一些同學(xué)的面經(jīng),發(fā)現(xiàn)無論什么技術(shù)崗位,還是會問到和的區(qū)別,而搜索出來的答案并不能讓我們裝得一手好逼,那就讓我們從報文的角度來擼一波,從而搞明白他們的區(qū)別。所以,和分開發(fā)送是部分瀏覽器或框架的請求方法,不屬于必然行為。
1 前言
最近看了一些同學(xué)的面經(jīng),發(fā)現(xiàn)無論什么技術(shù)崗位,還是會問到 get 和 post 的區(qū)別,而搜索出來的答案并不能讓我們裝得一手好逼,那就讓我們從 HTTP 報文的角度來擼一波,從而搞明白他們的區(qū)別。
2 標(biāo)準(zhǔn)答案在開擼之前嗎,讓我們先看一下標(biāo)準(zhǔn)答案長什么樣子 w3school: GET 對比 POST。標(biāo)準(zhǔn)答案很美好,但是在面試的時候把下面的表格甩面試官一臉,估計會裝逼不成反被*。
分類 | GET | POST |
---|---|---|
后退按鈕/刷新 | 無害 | 數(shù)據(jù)會被重新提交(瀏覽器應(yīng)該告知用戶數(shù)據(jù)會被重新提交)。 |
書簽 | 可收藏為書簽 | 不可收藏為書簽 |
緩存 | 能被緩存 | 不能緩存 |
編碼類型 | application/x-www-form-urlencoded | application/x-www-form-urlencoded 或 multipart/form-data。為二進制數(shù)據(jù)使用多重編碼。 |
歷史 | 參數(shù)保留在瀏覽器歷史中。 | 參數(shù)不會保存在瀏覽器歷史中。 |
對數(shù)據(jù)長度的限制 | 是的。當(dāng)發(fā)送數(shù)據(jù)時,GET 方法向 URL 添加數(shù)據(jù);URL 的長度是受限制的(URL 的最大長度是 2048 個字符)。 | 無限制。 |
對數(shù)據(jù)類型的限制 | 只允許 ASCII 字符。 | 沒有限制。也允許二進制數(shù)據(jù)。 |
安全性 | 與 POST 相比,GET 的安全性較差,因為所發(fā)送的數(shù)據(jù)是 URL 的一部分。在發(fā)送密碼或其他敏感信息時絕不要使用 GET ! | POST 比 GET 更安全,因為參數(shù)不會被保存在瀏覽器歷史或 web 服務(wù)器日志中。 |
可見性 | 數(shù)據(jù)在 URL 中對所有人都是可見的。 | 數(shù)據(jù)不會顯示在 URL 中。 |
注意,并不是說標(biāo)準(zhǔn)答案有誤,上述區(qū)別在大部分瀏覽器上是存在的,因為這些瀏覽器實現(xiàn)了 HTTP 標(biāo)準(zhǔn)。
所以從標(biāo)準(zhǔn)上來看,GET 和 POST 的區(qū)別如下:
GET 用于獲取信息,是無副作用的,是冪等的,且可緩存
POST 用于修改服務(wù)器上的數(shù)據(jù),有副作用,非冪等,不可緩存
但是,既然本文從報文角度來說,那就先不討論 RFC 上的區(qū)別,單純從數(shù)據(jù)角度談?wù)劇?/p> 3 GET 和 POST 報文上的區(qū)別
先下結(jié)論,GET 和 POST 方法沒有實質(zhì)區(qū)別,只是報文格式不同。
GET 和 POST 只是 HTTP 協(xié)議中兩種請求方式,而 HTTP 協(xié)議是基于 TCP/IP 的應(yīng)用層協(xié)議,無論 GET 還是 POST,用的都是同一個傳輸層協(xié)議,所以在傳輸上,沒有區(qū)別。
報文格式上,不帶參數(shù)時,最大區(qū)別就是第一行方法名不同
POST方法請求報文第一行是這樣的 POST /uri HTTP/1.1
GET方法請求報文第一行是這樣的 GET /uri HTTP/1.1
是的,不帶參數(shù)時他們的區(qū)別就僅僅是報文的前幾個字符不同而已
帶參數(shù)時報文的區(qū)別呢? 在約定中,GET 方法的參數(shù)應(yīng)該放在 url 中,POST 方法參數(shù)應(yīng)該放在 body 中
舉個例子,如果參數(shù)是 name=qiming.c, age=22。
GET 方法簡約版報文是這樣的
GET /index.php?name=qiming.c&age=22 HTTP/1.1 Host: localhost
POST 方法簡約版報文是這樣的
POST /index.php HTTP/1.1 Host: localhost Content-Type: application/x-www-form-urlencoded name=qiming.c&age=22
現(xiàn)在我們知道了兩種方法本質(zhì)上是 TCP 連接,沒有差別,也就是說,如果我不按規(guī)范來也是可以的。我們可以在 URL 上寫參數(shù),然后方法使用 POST;也可以在 Body 寫參數(shù),然后方法使用 GET。當(dāng)然,這需要服務(wù)端支持。
4. 常見問題 GET 方法參數(shù)寫法是固定的嗎?在約定中,我們的參數(shù)是寫在 ? 后面,用 & 分割。
我們知道,解析報文的過程是通過獲取 TCP 數(shù)據(jù),用正則等工具從數(shù)據(jù)中獲取 Header 和 Body,從而提取參數(shù)。
也就是說,我們可以自己約定參數(shù)的寫法,只要服務(wù)端能夠解釋出來就行,一種比較流行的寫法是 http://www.example.com/user/name/chengqm/age/22。
POST 方法比 GET 方法安全?按照網(wǎng)上大部分文章的解釋,POST 比 GET 安全,因為數(shù)據(jù)在地址欄上不可見。
然而,從傳輸?shù)慕嵌葋碚f,他們都是不安全的,因為 HTTP 在網(wǎng)絡(luò)上是明文傳輸?shù)模灰诰W(wǎng)絡(luò)節(jié)點上捉包,就能完整地獲取數(shù)據(jù)報文。
要想安全傳輸,就只有加密,也就是 HTTPS。
GET 方法的長度限制是怎么回事?在網(wǎng)上看到很多關(guān)于兩者區(qū)別的文章都有這一條,提到瀏覽器地址欄輸入的參數(shù)是有限的。
首先說明一點,HTTP 協(xié)議沒有 Body 和 URL 的長度限制,對 URL 限制的大多是瀏覽器和服務(wù)器的原因。
瀏覽器原因就不說了,服務(wù)器是因為處理長 URL 要消耗比較多的資源,為了性能和安全(防止惡意構(gòu)造長 URL 來攻擊)考慮,會給 URL 長度加限制。
POST 方法會產(chǎn)生兩個TCP數(shù)據(jù)包?有些文章中提到,post 會將 header 和 body 分開發(fā)送,先發(fā)送 header,服務(wù)端返回 100 狀態(tài)碼再發(fā)送 body。
HTTP 協(xié)議中沒有明確說明 POST 會產(chǎn)生兩個 TCP 數(shù)據(jù)包,而且實際測試(Chrome)發(fā)現(xiàn),header 和 body 不會分開發(fā)送。
所以,header 和 body 分開發(fā)送是部分瀏覽器或框架的請求方法,不屬于 post 必然行為。
5 talk is cheap show me the code如果對 get 和 post 報文區(qū)別有疑惑,直接起一個 Socket 服務(wù)端,然后封裝簡單的 HTTP 處理方法,直接觀察和處理 HTTP 報文,就能一目了然
#!/usr/bin/env python # -*- coding: utf-8 -*- import socket HOST, PORT = "", 23333 def server_run(): listen_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) listen_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) listen_socket.bind((HOST, PORT)) listen_socket.listen(1) print("Serving HTTP on port %s ..." % PORT) while True: # 接受連接 client_connection, client_address = listen_socket.accept() handle_request(client_connection) def handle_request(client_connection): # 獲取請求報文 request = "" while True: recv_data = client_connection.recv(2400) recv_data = recv_data.decode() request += recv_data if len(recv_data) < 2400: break # 解析首行 first_line_array = request.split(" ")[0].split(" ") # 分離 header 和 body space_line_index = request.index(" ") header = request[0: space_line_index] body = request[space_line_index + 4:] # 打印請求報文 print(request) # 返回報文 http_response = b""" HTTP/1.1 200 OKHello, World! Hello, World!
""" client_connection.sendall(http_response) client_connection.close() if __name__ == "__main__": server_run()
上面代碼就是簡單的打印請求報文然后返回 HelloWorld 的 html 頁面,我們運行起來
[root@chengqm shell]# python httpserver.py Serving HTTP on port 23333 ...
然后從瀏覽器中請求看看
打印出來的報文
然后就可以手動證明上述說法,比如說要測試 header 和 body 是否分開傳輸,由于代碼沒有返回 100 狀態(tài)碼,如果我們 post 請求成功就說明是一起傳輸?shù)?Chrome/postman)。
又比如 w3school 里面說 URL 的最大長度是 2048 個字符,那我們在代碼里面加上一句計算 uri 長度的代碼
... # 解析首行 first_line_array = request.split(" ")[0].split(" ") print("uri長度: %s" % len(first_line_array[1])) ...
我們用 postman 直接發(fā)送超過 2048 個字符的請求看看
然后我們可以得出結(jié)論,url 長度限制是某些瀏覽器和服務(wù)器的限制,和 HTTP 協(xié)議沒有關(guān)系。
到此,我們可以愉快地裝逼了 :)
參考:
99%的人都理解錯了HTTP中GET與POST的區(qū)別
關(guān)于HTTP GET 和 POST
w3school: HTTP 方法:GET 對比 POST
wikipedia: 超文本傳輸協(xié)議
RFC 2068
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://www.ezyhdfw.cn/yun/43166.html
摘要:前言最近看了一些同學(xué)的面經(jīng),發(fā)現(xiàn)無論什么技術(shù)崗位,還是會問到和的區(qū)別。所有學(xué)技術(shù)的同學(xué)都知道和函數(shù)怎么用,知道和的區(qū)別就是是儲存在服務(wù)端的,是存儲在瀏覽器的。的誕生是為了能讓無狀態(tài)的報文帶上一些特殊的數(shù)據(jù),讓服務(wù)端能夠辨識請求的身份。 1 前言 最近看了一些同學(xué)的面經(jīng),發(fā)現(xiàn)無論什么技術(shù)崗位,還是會問到 Session 和 Cookie 的區(qū)別。 所有學(xué)技術(shù)的同學(xué)都知道 Session ...
摘要:聯(lián)調(diào)測試,無需依賴他人。針對以上問題,有兩種解決方法,一個是自己搭建私有服務(wù),另一個是用云服務(wù)的鏡像管理平臺如阿里云的容器鏡像服務(wù)。利用,先對阿里云的服務(wù)進行登錄。推送后,就能在阿里云的倉庫上看到這個鏡像。 Docker簡述 Docker是一種OS虛擬化技術(shù),是一個開源的應(yīng)用容器引擎。它可以讓開發(fā)者將應(yīng)用打包到一個可移植的容器中,并且該容器可以運行在幾乎所有l(wèi)inux系統(tǒng)中(Windo...
摘要:示例使用作為深度,展開任意深度的嵌套數(shù)組會移除數(shù)組中的空項首先使用映射函數(shù)映射每個元素,然后將結(jié)果壓縮成一個新數(shù)組。注意對象數(shù)組不能使用方法來檢測。也就是返回值返回一個新的對象,該對象包含數(shù)組中每個索引的鍵值對。 showImg(https://segmentfault.com/img/remote/1460000019303737?w=651&h=289); 前言 寫久了業(yè)務(wù)代碼的我...
摘要:面試必考題吧,所以在這會詳細(xì)介紹以下內(nèi)容跨域產(chǎn)生的原因羅列最常用的解決方法分析各種方法原理羅列各種方法優(yōu)缺點什么是跨域由于瀏覽器廠商對安全性的考慮,提出了瀏覽器的同源策略做為解決方案。 面試必考題吧,所以在這會詳細(xì)介紹以下內(nèi)容 跨域產(chǎn)生的原因 羅列最常用的解決方法 分析各種方法原理 羅列各種方法優(yōu)缺點 什么是跨域 由于瀏覽器廠商對安全性的考慮,提出了瀏覽器的同源策略做為解決方案。它...
摘要:面試必考題吧,所以在這會詳細(xì)介紹以下內(nèi)容跨域產(chǎn)生的原因羅列最常用的解決方法分析各種方法原理羅列各種方法優(yōu)缺點什么是跨域由于瀏覽器廠商對安全性的考慮,提出了瀏覽器的同源策略做為解決方案。 面試必考題吧,所以在這會詳細(xì)介紹以下內(nèi)容 跨域產(chǎn)生的原因 羅列最常用的解決方法 分析各種方法原理 羅列各種方法優(yōu)缺點 什么是跨域 由于瀏覽器廠商對安全性的考慮,提出了瀏覽器的同源策略做為解決方案。它...
閱讀 2764·2023-04-26 02:47
閱讀 3129·2023-04-26 00:42
閱讀 1003·2021-10-12 10:12
閱讀 1524·2021-09-29 09:35
閱讀 1857·2021-09-26 09:55
閱讀 613·2019-08-30 14:00
閱讀 1654·2019-08-29 12:57
閱讀 2480·2019-08-28 18:00