對Golang中的FORM相關字段理解
Form 字段
通過調用Request結構體提供的方法,我們可以將URL、Body、或者以上兩者的數據提取到該結構體的Form、PostForm和MultipartForm等字段中。
(1)調用ParseForm方法或者ParseMultipartForm方法,對請求進行分析
(2)訪問相應的字段
事例:
package main import ( "net/http" "fmt" ) func process(w http.ResponseWriter, r *http.Request) { r.ParseForm() //ParseForm 對請求進行語法分析 fmt.Fprintln(w,r.MultipartForm) } func main() { server := http.Server{ Addr:"127.0.0.1:8080", } http.HandleFunc("/process",process) server.ListenAndServe() }
創建一個具體表單
<!DOCTYPE html> <html> <head> <meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> <title>GoWebPrograming</title> </head> <body> <form action="http://127.0.0.1:8080/process?hello=world&thread=get" method="post" enctype="application/x-www-form-urlencoded"> <input type="text" name="hello" value="你好 世界"/> <input type="text" name="post" value="456" /> <input type="submit" /> </form> </body> </html>
我們在瀏覽器運行html文件,結果為:
map[hello:[你好 世界 world] post:[456] thread:[get]]
我們發現這個結構是一個map,他的鍵為字符串,而建的值是由字符串組成的一個切片。
這個結構總是包含查詢的值hello=world, thread=get,還有表單值hello=123和post=456,這些值都進行瞭url的解碼。
比如你好世界之間有空格,說明不是編碼之後的%20。
PostForm 字段
執行語句r.Form[“post”]會返回一個切片,切片裡包含瞭表單提交的數據和url中的數據就像“你好世界”和“world” 是一組切片值。但是表單值在切片中總會排在url之前。 ( hello:[你好 世界 world] )
如果我們隻想獲得表單值而不是url的值,我們可以使用Request結構的PostForm字段,
我們將r.Form 改為 r.PostForm 會出現如下結果
map[hello:[你好 世界] post:[456]]
我們將 enctype=”application/x-www-form-urlencoded”改為 enctype=“multipart/form-data”, 結果如下:
map[]
會得到一個空的map,這是為什麼呢???
如果我們將 enctype=”application/x-www-form-urlencoded”改為 enctype=“multipart/form-data”,並改回 r.Form。會出現以下結果:
map[hello:[world] thread:[get]]
這是因為ParseForm字段隻支持”application/x-www-form-urlencoded”編碼,所以r.Form不會反悔任何表單值,而是隻返回url的查詢值。
為瞭解決這個問題,我們需要通過MultipartForm字段來獲取multipart/form-data編碼的表單值。
補充:go通過http發送form-data
首先是獲取form-data內容
func ResendFormFile(r *http.Request, URL string) { data := r.FormValue("data") formFile, fileHeader, err := r.FormFile("pic") if err != nil { return } _, status := RequestPost(formFile, fileHeader.Filename, []byte(data), URL) if (status / 100) != 2 { fmt.Println("轉發圖片失敗") } return }
然後是發送
func RequestPost(formFile multipart.File, filename string, data []byte, postURL string) (resp interface{}, status int) { buf := new(bytes.Buffer) w := multipart.NewWriter(buf) if fw, err := w.CreateFormField("data"); err == nil { fw.Write(data) } if createFormFile, err := w.CreateFormFile("pic", filename); err == nil { readAll, _ := ioutil.ReadAll(formFile) createFormFile.Write(readAll) } w.Close() req, err := http.NewRequest(http.MethodPost, postURL, buf) if err != nil { return } // Don't forget to set the content type, this will contain the boundary. req.Header.Set("Content-Type", w.FormDataContentType()) client := &http.Client{} res, err := client.Do(req) if err != nil { return } return res.Body, res.StatusCode }
這樣返回的body是不可以直接json序列化的
可以先使用ioutil讀出來或者byte.Buffer進行中轉都是比較簡單的選擇
func UnmarshalWriter(body io.ReadCloser, w http.ResponseWriter) { all, _ := ioutil.ReadAll(body) buffer := bytes.NewBuffer(all) buffer.WriteTo(w) }
以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。如有錯誤或未考慮完全的地方,望不吝賜教。
推薦閱讀:
- Go語言實現文件上傳
- GO實現文件上傳操作
- php使用file_get_contents(‘php://input‘)和$_POST的區別實例對比
- 使用Spring處理x-www-form-urlencoded方式
- 使用flask如何獲取post請求參數