1. 程式人生 > 其它 >http get請求_黑客入門:HTTP方法——GET、POST

http get請求_黑客入門:HTTP方法——GET、POST

技術標籤:http get請求http post請求http報文格式http報文格式分析http方法http請求必須在body中傳遞引數值

什麼是 HTTP?

超文字傳輸協議(HTTP)的設計目的是保證客戶機與伺服器之間的通訊。

HTTP 的工作方式是客戶機與伺服器之間的請求-應答協議。

web 瀏覽器可能是客戶端,而計算機上的網路應用程式也可能作為伺服器端。

舉例:客戶端(瀏覽器)向伺服器提交 HTTP 請求;伺服器向客戶端返回響應。響應包含關於請求的狀態資訊以及可能被請求的內容。

這也是一種基於C/S的開發模型,又稱客戶端(client)/伺服器(server)模型

cc42d8b5786103c82a12e440830630be.png

HTTP請求報文格式:

d5cec5085f887b34e0969920ecd6f8aa.png

(圖片來源於網路)

HTTP響應報文格式:

39277d9d7b81f81f96f5e3caa6e31e0f.png

(圖片來源於網路)

重要的兩種 HTTP 請求方法:GET 和 POST

在客戶機和伺服器之間進行請求-響應時,兩種最常被用到的方法是:GET 和 POST。此外,有興趣可以瞭解一下head方法、option方法、PUT方法,這三種方法在iis6的檔案上傳漏洞中會用到。

HTTP的底層是TCP/IP。所以GET和POST的底層也是TCP/IP,也就是說,GET/POST都是TCP連結。GET和POST能做的事情是一樣一樣的。你要給GET加上request body,給POST帶上url引數,技術上是完全行的通的。

  • GET - 從指定的資源請求資料。獲取資訊,是無副作用的,是冪等的,且可快取。

  • POST - 向指定的資源提交要被處理的資料。修改伺服器上的資料,有副作用,非冪等,不可快取。

GET 方法

請注意,查詢字串(名稱/值對)是在 GET請求的URL中傳送的:

例如一段url:

http://xx.com/test/demo_form.asp?name1=value1&name2=value2,

我們可以看到客戶GET了兩個引數

引數1:name1 引數1的值:value1

引數2:name2 引數2的值:value2

其中http是請求協議,與此類似的還有ftp、samba、https等,這些都是應用層協議。

xx.com是域名,域名和ip通過dns服務形成對映,有關dns劫持的問題,可以檢視公眾號中的歷史文章。

/test/demo_form.asp是代表了基於web伺服器的web應用程式的相對路徑,這代表我們正在訪問的web資源。

有關 GET 請求的其他一些註釋:

  • GET 請求可被快取

  • GET 請求保留在瀏覽器歷史記錄中

  • GET 請求可被收藏為書籤

  • GET 請求不應在處理敏感資料時使用

  • GET 請求有長度限制

  • GET 請求只應當用於取回資料

POST 方法

請注意,查詢字串(名稱/值對)是在POST 請求的 HTTP 訊息主體中傳送的:

POST /test/demo_form.asp HTTP/1.1
Host: w3schools.com

name1=value1&name2=value2

這需要通過檢視http報文來分析,不能直接通過url直觀的展現。

有關 POST 請求的其他一些註釋:

  • POST 請求不會被快取

  • POST 請求不會保留在瀏覽器歷史記錄中

  • POST 不能被收藏為書籤

  • POST 請求對資料長度沒有要求

GET 和 POST 報文上的區別

先下結論,GET 和 POST 方法沒有實質區別,只是報文格式不同。

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

報文格式上,不帶引數時,最大區別就是第一行方法名不同

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

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

是的,不帶引數時他們的區別就僅僅是報文的前幾個字元不同而已

帶引數時報文的區別呢?在約定中,GET 方法的引數應該放在 url 中,POST 方法引數應該放在 body 中

舉個例子,如果引數是 name=ooxx, age=18。

GET 方法簡約版報文是這樣的

GET /index.php?name=ooxx&age=18 HTTP/1.1Host: localhost

POST 方法簡約版報文是這樣的

POST /index.php HTTP/1.1Host: localhostContent-Type: application/x-www-form-urlencoded
name=ooxx&age=18

現在我們知道了兩種方法本質上是 TCP 連線,沒有差別,也就是說,如果我不按規範來也是可以的。我們可以在 URL 上寫引數,然後方法使用 POST;也可以在 Body 寫引數,然後方法使用 GET。當然,這需要服務端支援。

常見問題

GET 方法引數寫法是固定的嗎?

在約定中,我們的引數是寫在?後面,用&分割。

我們知道,解析報文的過程是通過獲取 TCP 資料,用正則等工具從資料中獲取 Header 和 Body,從而提取引數。

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

POST 方法比 GET 方法安全?

按照網上大部分文章的解釋,POST 比 GET 安全,因為資料在位址列上不可見。

然而,從傳輸的角度來說,他們都是不安全的,因為 HTTP 在網路上是明文傳輸的,只要在網路節點上捉包,就能完整地獲取資料報文。

要想安全傳輸,就只有加密,也就是 HTTPS。

GET 方法的長度限制是怎麼回事?

在網上看到很多關於兩者區別的文章都有這一條,提到瀏覽器位址列輸入的引數是有限的。

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

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

(GET方法因此最多傳輸256個字元長度)

POST 方法會產生兩個 TCP 資料包?

有些文章中提到,post 會將 header 和 body 分開發送,先發送 header,服務端返回 100 狀態碼再發送 body。

HTTP 協議中沒有明確說明 POST 會產生兩個 TCP 資料包,而且實際測試(Chrome)發現,header 和 body 不會分開發送。

所以,header 和 body 分開發送是部分瀏覽器或框架的請求方法,不屬於 post 必然行為。

GET和POST的差別:

  • GET在瀏覽器回退時是無害的,而POST會再次提交請求。

  • GET產生的URL地址可以被Bookmark,而POST不可以。

  • GET請求會被瀏覽器主動cache,而POST不會,除非手動設定。

  • GET請求只能進行url編碼,而POST支援多種編碼方式。

  • GET請求引數會被完整保留在瀏覽器歷史記錄裡,而POST中的引數不會被保留。

  • GET請求在URL中傳送的引數是有長度限制的,而POST麼有。

  • 對引數的資料型別,GET只接受ASCII字元,而POST沒有限制。

  • GET比POST更不安全,因為引數直接暴露在URL上,所以不能用來傳遞敏感資訊。

  • GET引數通過URL傳遞,POST放在Request body中。