日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

Http / Get 和 Post 区别

發(fā)布時間:2024/10/14 编程问答 37 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Http / Get 和 Post 区别 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

一、前言

最近看了一些同學的面經(jīng),發(fā)現(xiàn)無論什么技術崗位,還是會問到 get 和 post 的區(qū)別,而搜索出來的答案并不能讓我們裝得一手好逼,那就讓我們從 HTTP 報文的角度來擼一波,從而搞明白他們的區(qū)別。

二、標準答案

在開擼之前嗎,讓我們先看一下標準答案長什么樣子 w3school: GET 對比 POST。標準答案很美好,但是在面試的時候把下面的表格甩面試官一臉,估計會裝逼不成反被*。

分類GETPOST
后退按鈕/刷新無害數(shù)據(jù)會被重新提交(瀏覽器應該告知用戶數(shù)據(jù)會被重新提交)。
書簽可收藏為書簽不可收藏為書簽
緩存能被緩存不能緩存
編碼類型application/x-www-form-urlencodedapplication/x-www-form-urlencoded 或 multipart/form-data。為二進制數(shù)據(jù)使用多重編碼。
歷史參數(shù)保留在瀏覽器歷史中。參數(shù)不會保存在瀏覽器歷史中。
對數(shù)據(jù)長度的限制是的。當發(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 服務器日志中。
可見性數(shù)據(jù)在 URL 中對所有人都是可見的。數(shù)據(jù)不會顯示在 URL 中。

注意,并不是說標準答案有誤,上述區(qū)別在大部分瀏覽器上是存在的,因為這些瀏覽器實現(xiàn)了 HTTP 標準。

所以從標準上來看,GET 和 POST 的區(qū)別如下:

  • GET 用于獲取信息,是無副作用的,是冪等的,且可緩存。
  • POST 用于修改服務器上的數(shù)據(jù),有副作用,非冪等,不可緩存。

但是,既然本文從報文角度來說,那就先不討論 RFC 上的區(qū)別,單純從數(shù)據(jù)角度談談。

三、GET 和 POST 報文上的區(qū)別

先下結(jié)論:GET 和 POST 方法沒有實質(zhì)區(qū)別,只是報文格式不同。

GET 和 POST 只是 HTTP 協(xié)議中兩種請求方式,而 HTTP 協(xié)議是基于 TCP/IP 的應用層協(xié)議,無論 GET 還是 POST,用的都是同一個傳輸層協(xié)議,所以在傳輸上,沒有區(qū)別。

報文格式上,不帶參數(shù)時,最大區(qū)別就是第一行方法名不同:

POST方法請求報文第一行是這樣的 POST /uri HTTP/1.1 \r\n

GET方法請求報文第一行是這樣的 GET /uri HTTP/1.1 \r\n

是的,不帶參數(shù)時他們的區(qū)別就僅僅是報文的前幾個字符不同而已。

帶參數(shù)時報文的區(qū)別呢? 在約定中,GET 方法的參數(shù)應該放在 url 中,POST 方法參數(shù)應該放在 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-urlencodedname=qiming.c&age=22

現(xiàn)在我們知道了兩種方法本質(zhì)上是 TCP 連接,沒有差別,也就是說,如果我不按規(guī)范來也是可以的。我們可以在 URL 上寫參數(shù),然后方法使用 POST;也可以在 Body 寫參數(shù),然后方法使用 GET。當然,這需要服務端支持。

四、常見問題

1、GET 方法參數(shù)寫法是固定的嗎?

在約定中,我們的參數(shù)是寫在 ? 后面,用 & 分割。

我們知道,解析報文的過程是通過獲取 TCP 數(shù)據(jù),用正則等工具從數(shù)據(jù)中獲取 Header 和 Body,從而提取參數(shù)。

也就是說,我們可以自己約定參數(shù)的寫法,只要服務端能夠解釋出來就行,一種比較流行的寫法是 http://www.example.com/user/name/chengqm/age/22。

2、POST 方法比 GET 方法安全?

按照網(wǎng)上大部分文章的解釋,POST 比 GET 安全,因為數(shù)據(jù)在地址欄上不可見。

然而,從傳輸?shù)慕嵌葋碚f,他們都是不安全的,因為 HTTP 在網(wǎng)絡上是明文傳輸?shù)?#xff0c;只要在網(wǎng)絡節(jié)點上捉包,就能完整地獲取數(shù)據(jù)報文。要想安全傳輸,就只有加密,也就是 HTTPS。

3、GET 方法的長度限制是怎么回事?

在網(wǎng)上看到很多關于兩者區(qū)別的文章都有這一條,提到瀏覽器地址欄輸入的參數(shù)是有限的。

首先說明一點,HTTP 協(xié)議沒有 Body 和 URL 的長度限制,對 URL 限制的大多是瀏覽器和服務器的原因。

瀏覽器原因就不說了,服務器是因為處理長 URL 要消耗比較多的資源,為了性能和安全(防止惡意構(gòu)造長 URL 來攻擊)考慮,會給 URL 長度加限制。

4、POST 方法會產(chǎn)生兩個TCP數(shù)據(jù)包?

有些文章中提到,post 會將 header 和 body 分開發(fā)送,先發(fā)送 header,服務端返回 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 服務端,然后封裝簡單的 HTTP 處理方法,直接觀察和處理 HTTP 報文,就能一目了然

#!/usr/bin/env python # -*- coding: utf-8 -*-import socketHOST, PORT = '', 23333def 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_dataif len(recv_data) < 2400:break# 解析首行first_line_array = request.split('\r\n')[0].split(' ')# 分離 header 和 bodyspace_line_index = request.index('\r\n\r\n')header = request[0: space_line_index]body = request[space_line_index + 4:]# 打印請求報文print(request)# 返回報文http_response = b"""\ HTTP/1.1 200 OK<!DOCTYPE html> <html> <head><title>Hello, World!</title> </head> <body> <p style="color: green">Hello, World!</p> </body> </html> """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('\r\n')[0].split(' ') print('uri長度: %s' % len(first_line_array[1])) ...

我們用 postman 直接發(fā)送超過 2048 個字符的請求看看

然后我們可以得出結(jié)論,url 長度限制是某些瀏覽器和服務器的限制,和 HTTP 協(xié)議沒有關系。

到此,我們可以愉快地裝逼了 :)

?

轉(zhuǎn)載于:https://segmentfault.com/a/1190000018129846(若侵則刪)

?

(SAW:Game Over!)

總結(jié)

以上是生活随笔為你收集整理的Http / Get 和 Post 区别的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。