成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

都2019年了,還問GET和POST的區別

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

[[257239]]

1.前言

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

2. 標準答案

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

分類 GET POST
后退按鈕/刷新 無害 數據會被重新提交(瀏覽器應該告知用戶數據會被重新提交)。
書簽 可收藏為書簽 不可收藏為書簽
緩存 能被緩存 不能緩存
編碼類型 application/x-www-form-urlencoded application/x-www-form-urlencoded 或 multipart/form-data。為二進制數據使用多重編碼。
歷史 參數保留在瀏覽器歷史中。 參數不會保存在瀏覽器歷史中。
對數據長度的限制 是的。當發送數據時,GET 方法向 URL 添加數據;URL 的長度是受限制的(URL 的最大長度是 2048 個字符)。 無限制。
對數據類型的限制 只允許 ASCII 字符。 沒有限制。也允許二進制數據。
安全性 與 POST 相比,GET 的安全性較差,因為所發送的數據是 URL 的一部分。在發送密碼或其他敏感信息時絕不要使用 GET ! POST 比 GET 更安全,因為參數不會被保存在瀏覽器歷史或 web 服務器日志中。
可見性 數據在 URL 中對所有人都是可見的。 數據不會顯示在 URL 中。

注意,并不是說標準答案有誤,上述區別在大部分瀏覽器上是存在的,因為這些瀏覽器實現了 HTTP 標準。但是,前面列舉的只是瀏覽器實現上的區別,而不是 get 和 post 的本質區別。

3.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=chengqm, age=22。

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

  1. GET /index.php?name=qiming.c&age=22 HTTP/1.1  
  2. Host: localhost  

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

  1. POST /index.php HTTP/1.1  
  2. Host: localhost  
  3. Content-Type: application/x-www-form-urlencoded  
  4. name=qiming.c&age=22  

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

4. 常見問題

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 長度加限制。

POST 方法會產生兩個TCP數據包?

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

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

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

5. talk is cheap show me the code

如果對 get 和 post 報文區別有疑惑,直接起一個 Socket 服務端,然后封裝簡單的 HTTP 處理方法,直接觀察和處理 HTTP 報文,就能一目了然。 

  1. #!/usr/bin/env python  
  2. # -*- coding: utf-8 -*-  
  3. import socket  
  4. HOST, PORT = '', 23333  
  5. def server_run():  
  6. listen_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)  
  7. listen_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)  
  8. listen_socket.bind((HOST, PORT)) 
  9. listen_socket.listen(1)  
  10. print('Serving HTTP on port %s ...' % PORT)  
  11. while True 
  12. # 接受連接  
  13. client_connection, client_address = listen_socket.accept()  
  14. handle_request(client_connection)  
  15. def handle_request(client_connection):  
  16. # 獲取請求報文  
  17. request = ''  
  18. while True 
  19. recv_data = client_connection.recv(2400)  
  20. recv_data = recv_data.decode()  
  21. request += recv_data  
  22. if len(recv_data) < 2400:  
  23. break  
  24. # 解析首行  
  25. first_line_array = request.split('\r\n')[0].split(' ' 
  26. # 分離 header 和 body  
  27. space_line_index = request.index('\r\n\r\n' 
  28. header = request[0: space_line_index]  
  29. body = request[space_line_index + 4:]  
  30. # 打印請求報文  
  31. print(request)  
  32. # 返回報文  
  33. http_response = b"""\  
  34. HTTP/1.1 200 OK  
  35. Hello, World!  
  36. "" 
  37. client_connection.sendall(http_response)  
  38. client_connection.close()  
  39. if __name__ == '__main__' 
  40. server_run()  

上面代碼就是簡單的打印請求報文然后返回 HelloWorld 的 html 頁面,我們運行起來。 

  1. [root@chengqm shell]# python httpserver.py  
  2. Serving HTTP on port 23333 ...  

然后從瀏覽器中請求看看: 

打印出來的報文: 

 

然后就可以手動證明上述說法,比如說要測試 header 和 body 是否分開傳輸,由于代碼沒有返回 100 狀態碼,如果我們 post 請求成功就說明是一起傳輸的(Chrome/postman)。 

 

又比如 w3school 里面說 URL 的最大長度是 2048 個字符,那我們在代碼里面加上一句計算 uri 長度的代碼。 

  1. ...  
  2. # 解析首行  
  3. first_line_array = request.split('\r\n')[0].split(' ' 
  4. print('uri長度: %s' % len(first_line_array[1]))  
  5. ...  

我們用 postman 直接發送超過 2048 個字符的請求看看。 

 

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

責任編輯:龐桂玉 來源: segmentfault
相關推薦

2011-06-08 11:02:14

GetPost

2023-08-07 09:25:06

GETPOSTHTTP請求

2019-05-15 16:15:08

HTTPGETPOST

2016-09-23 15:10:10

HTTPGETPOST

2019-08-08 16:00:08

HTTPGETPOST

2022-08-03 07:04:56

GETHTTPPOST

2011-06-08 11:05:38

getpost

2020-09-24 10:09:01

GETPOST前端

2014-07-24 10:00:21

2023-01-30 15:39:40

GETHTTP

2009-07-29 16:53:52

ASP.NET Get

2009-06-29 18:32:30

Hibernate

2019-06-05 10:23:00

2020-12-09 09:22:53

GETPOSTWeb

2009-06-26 16:23:12

Hibernate gHibernate l

2021-11-30 08:57:43

GETPOSTJava

2009-09-28 13:23:54

HTTP編程

2017-10-31 08:43:14

2015-08-06 13:33:22

PHPGETPOST

2015-10-27 11:06:51

PHPGETPOST
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产一区二区三区四区区 | 亚洲精品日韩一区二区电影 | 免费国产一区 | 日韩国产一区 | 91精品国产一区二区三区蜜臀 | 亚洲三级av | 国产一区二区在线视频 | 国外成人免费视频 | 久久网亚洲| 九九亚洲 | 日韩三片 | 99re在线 | 国产69久久精品成人看动漫 | 国产精品美女久久久 | 亚洲午夜精品 | 午夜激情一区 | 草草视频在线观看 | 久久久久久天堂 | 国产综合精品一区二区三区 | 久久久久久久91 | 日本不卡一区二区三区在线观看 | 一二三在线视频 | 狠狠综合网 | 亚洲三区在线 | 阿v视频在线观看 | 丝袜美腿一区二区三区动态图 | 美女国产精品 | 日韩一区二区三区av | 在线午夜 | 国产中文字幕网 | 国产日产精品一区二区三区四区 | 欧美成年网站 | 免费观看一级特黄欧美大片 | 欧美精品久久久久久久久老牛影院 | a视频在线观看 | 超碰网址 | 毛片黄片免费看 | 天天操操 | 欧美日韩综合一区 | www.国产 | 国产精品久久久亚洲 |