goweb- 對請求的處理


對請求的處理

Go 語言的 net/http 包提供了一系列用於表示 HTTP 報文的結構,我們可以使用它
處理請求和發送相應,其中 Request 結構代表了客戶端發送的請求報文,下面讓我們看
一下 Request 結構體

獲取請求 URL

Request 結構中的 URL 字段用於表示請求行中包含的 URL,改字段是一個指向
url.URL 結構的指針

Path 字段

注: 通過 r.URL.Path 只能得到 /hello

RawQuery 字段

注: 通過 r.URL.RawQuery 得到的是 username=admin&password=123456

獲取請求頭中的信息

通過 Request 結果中的 Header 字段用來獲取請求頭中的所有信息,Header 字段
的類型是 Header 類型,而 Header 類型是一個 map[string][]string,string 類型的 key,
string 切片類型的值。

獲取請求頭中的所有信息

  • r.Header

  • 得到的結果如下:

map[User-Agent:[Mozilla/5.0 (Windows NT 10.0; Win64; x64)
AppleWebKit/537.36 (KHTML, like Gecko) Chrome/67.0.3396.62 Safari/537.36]
Accept:[text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,ima
ge/apng,*/*;q=0.8] Accept-Encoding:[gzip, deflate, br] Accept-Language:[zh-
CN,zh;q=0.9,en-US;q=0.8,en;q=0.7] Connection:[keep-alive] Upgrade-Insecure-
Requests:[1]]

獲取請求頭中的某個具體屬性的值,如獲取 Accept-Encoding 的值

  • 方式一:r.Header[“Accept-Encoding”]

    • 得到的是一個字符串切片
    • 結果[gzip, deflate, br]
  • 方式二:r.Header.Get(“Accept-Encoding”)

    • 得到的是字符串形式的值,多個值使用逗號分隔
    • 結果 gzip, deflate, br

獲取請求體中的信息

請求和響應的主體都是有 Request 結構中的 Body 字段表示,這個字段的類型是
io.ReadCloser 接口,該接口包含了 Reader 接口和 Closer 接口,Reader 接口擁有 Read
方法,Closer 接口擁有 Close 方法

通過指定 method=”post”來發送一個 POST 請求

由於 GET 請求沒有請求體,所以我們需要在 HTML 頁面中創建一個 form 表單,通
過指定 method=”post”來發送一個 POST 請求

  • 表單
<html>
  <head>
    <meta charset="UTF-8" />
  </head>
  <body>
    <form
      action="http://localhost:8080/getBody"
      method="POST"
      enctype="multipart/form-data"
    >
      用戶名:<input type="text" name="username" /><br />
      密碼:<input type="password" name="password" /><br />
      <input type="submit" />
    </form>
  </body>
</html>
  • 服務器處理請求的代碼
package main
import (
"fmt"
"net/http"
)
func handler(w http.ResponseWriter, r *http.Request) {
//獲取內容的長度
length := r.ContentLength
//創建一個字節切片
body := make([]byte, length)
//讀取請求體
r.Body.Read(body)
fmt.Fprintln(w, "請求體中的內容是:", string(body))
}
func main() {
http.HandleFunc("/getBody", handler)
http.ListenAndServe(":8080", nil)
}
  • 在瀏覽器上顯示的結果

    • 請求體中的內容是: username=hanzong&password=666666

獲取請求參數

下面我們就通過 net/http 庫中的 Request 結構的字段以及方法獲取請求 URL 后面
的請求參數以及 form 表單中提交的請求參數

Form 字段

  1. 類型是 url.Values 類型,Form 是解析好的表單數據,包括 URL 字段的 query
    參數和 POST 或 PUT 的表單數據。
  2. Form 字段只有在調用 Request 的 ParseForm 方法后才有效。在客戶端,會忽
    略請求中的本字段而使用 Body 替代
  3. 獲取表單中提交的請求參數(username 和 password)
  • 代碼
func handler(w http.ResponseWriter, r *http.Request) {
//解析表單
r.ParseForm()
//獲取請求參數
fmt.Fprintln(w, "請求參數為:", r.Form)
}
//注意:在執行 r.Form 之前一定要調用 ParseForm 方法
//結果
//請求參數為: map[password:[666666] username:[hanzong]]
  • d) 如果對 form 表單做一些修改,在 action 屬性的 URL 后面也添加相同的請求參
    數,如下:
<form
  action="http://localhost:8080/getBody?username=admin&pwd=123456"
  method="POST"
>
  用 戶 名 : <input type="text" name="username" value="hanzong" /><br />
  密 碼 : <input type="password" name="password" value="666666" /><br />
  <input type="submit" />
</form>

則執行結果如下:

請求參數為:map[username:[hanzong admin] password:[666666] pwd:[123456]]

  • 我們發現:表單中的請求參數 username 和 URL 中的請求參數
    username 都獲取到了,而且表單中的請求參數的值排在 URL 請求參
    數值的前面
  • 如果此時我們只想獲取表單中的請求參數該怎么辦呢?那就需要使
    用 Request 結構中的 PostForm 字段

PostForm 字段

  1. 類型也是 url.Values 類型,用來獲取表單中的請求參數
  • 將 r.Form 改為 r.PostForm 之后的代碼
func handler(w http.ResponseWriter, r *http.Request) {
//解析表單
r.ParseForm()
//獲取請求參數
fmt.Fprintln(w, "請求參數為:", r.PostForm)
}
  • 結果

請求參數為:map[username:[hanzong] password:[666666]]

  1. 但是 PostForm 字段只支持 application/x-www-form-urlencoded 編碼,如果
    form 表單的 enctype 屬性值為 multipart/form-data,那么使用 PostForm 字段
    無法獲取表單中的數據,此時需要使用 MultipartForm 字段
  • 說明:form 表單的 enctype 屬性的默認值是 application/x-www-form-
    urlencoded 編 碼 , 實 現 上 傳 文 件 時 需 要 講 該 屬 性 的 值 設 置 為
    multipart/form-data 編碼格式

FormValue 方法和 e PostFormValue 方法

FormValue 方法

a) 可以通過 FormValue 方法快速地獲取某一個請求參數,該方法調用之前
會自動調用 ParseMultipartForm 和 ParseForm 方法對表單進行解析

  • 代碼
func handler(w http.ResponseWriter, r *http.Request) {
//獲取請求參數
fmt.Fprintln(w, "請求參數username的值為:", r.FormValue("username"))
}

  • 結果

請求參數 username 的值為: hanzong

PostFormValue 方法

a) 可以通過 PostFormValue 方法快速地獲取表單中的某一個請求參數,該
方法調用之前會自動調用 ParseMultipartForm 和 ParseForm 方法對表單
進行解析

  • 代碼
func handler(w http.ResponseWriter, r *http.Request) {
//獲取請求參數
fmt.Fprintln(w, "請求參數 username 的值為:", r.PostFormValue("username"))
}
  • 結果

請求參數 username 的值為: hanzong

MultipartForm 字段

為了取得 multipart/form-data 編碼的表單數據,我們需要用到 Request 結構的
ParseMultipartForm 方法和 MultipartForm 字段,我們通常上傳文件時會將 form 表單的
enctype 屬性值設置為 multipart/form-data

  1. 表單
<form
  action="http://localhost:8080/upload"
  method="POST"
  enctype="multipart/form-data"
>
  用 戶 名 : <input type="text" name="username" value="hanzong" /><br />
  文件:<input type="file" name="photo" /><br />
  <input type="submit" />
</form>
  1. 后台處理請求代碼
package main
import (
"fmt"
"net/http"
)
func handler(w http.ResponseWriter, r *http.Request) {
//解析表單
r.ParseMultipartForm(1024)

fmt.Fprintln(w, r.MultipartForm)
}
func main() {
http.HandleFunc("/upload", handler)
http.ListenAndServe(":8080", nil)
}
  1. 瀏覽器顯示結果

&{map[username:[hanzong]] map[photo:[0xc042126000]]}

  • 結果中有兩個映射,第一個映射映射的是用戶名;第二個映射的值是一個地址

  • MuiltipartForm 字段的類型為 *multipart.Form,multipart 包下 Form 結構的指
    針類型

  • 打開上傳的文件

func handler(w http.ResponseWriter, r *http.Request) {
//解析表單
r.ParseMultipartForm(1024)
fileHeader := r.MultipartForm.File["photo"][0]
file, err := fileHeader.Open()
if err == nil {
data, err := ioutil.ReadAll(file)
if err == nil {
fmt.Fprintln(w, string(data))
}
}
}

FormFile 方法

  • net/http 提供的 FormFile 方法可以快速的獲取被上傳的文件,但是只能處理上
    傳一個文件的情況。

  • 代碼

func handler(w http.ResponseWriter, r *http.Request) {
file, _, err := r.FormFile("photo")
if err == nil {
data, err := ioutil.ReadAll(file)

if err == nil {
fmt.Fprintln(w, string(data))
}
}
}

給客戶端響應

前面我們一直說的是如何使用處理器中的 *http.Request 處理用戶的請求,下面我
們來說一下如何使用 http.ResponseWriter 來給用戶響應

給客戶端響應一個字符串

  • 處理器中的代碼
func handler(w http.ResponseWriter, r *http.Request) {
w.Write([]byte("你的請求我已經收到"))
}
  • 瀏覽器中的結果
    你的請求我已經收到
  • 響應報文中的內容
 HTTP/1.1 200 OK
 Date: Fri, 10 Aug 2019 01:09:27 GMT
 Content-Length: 27
 Content-Type: text/plain; charset=utf-8

  1. 給客戶端響應一個 HTML 頁面
  2. 處理器中的代碼
   func handler(w http.ResponseWriter, r \*http.Request) {
   html := `<html>
   <head>
   <title>測試響應內容為網頁</title>
   <meta charset="utf-8"/>
   </head>
   <body>
   我是以網頁的形式響應過來的!
   </body>
   </html>`
   w.Write([]byte(html))
   }

  1. 瀏覽器中的結果
    我是以網頁的形式響應過來的!
  • 通過在瀏覽器中右鍵 → 查看網頁代碼發現確實是一個 html 頁面
  1. 響應報文中的內容
   HTTP/1.1 200 OK
   Date: Fri, 10 Aug 2018 01:26:58 GMT
   Content-Length: 194
   Content-Type: text/html; charset=utf-8

  1. 給客戶端響應 JSON 格式的數據

  2. 處理器端代碼

   func handler(w http.ResponseWriter, r \*http.Request) {
   //設置響應頭中內容的類型
   w.Header().Set("Content-Type", "application/json")
   user := User{
   ID: 1,
   Username: "admin",
   Password: "123456",
   }
   //將 user 轉換為 json 格式
   json, \_ := json.Marshal(user)
   w.Write(json)
   }

  1. 瀏覽器中的結果
    {"ID":1,"Username":"admin","Password":"123456"}
  2. 響應報文中的內容
   HTTP/1.1 200 OK
   Content-Type: application/json
   Date: Fri, 10 Aug 2018 01:58:02 GMT
   Content-Length: 47

  1. 讓客戶端重定向

  2. 處理器端代碼

   func handler(w http.ResponseWriter, r \*http.Request) {
   //以下操作必須要在 WriteHeader 之前進行
   w.Header().Set("Location", "https:www.baidu.com")
   w.WriteHeader(302)
   }

  1. 響應報文中的內容
    HTTP/1.1 302 Found
    Location: https:www.baidu.com
    Date: Fri, 10 Aug 2018 01:45:04 GMT
    Content-Length: 0
    Content-Type: text/plain; charset=utf-8


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM