目錄html
更新、更全的《Go從入門到放棄》的更新網站,更有python、go、人工智能教學等着你:http://www.javashuo.com/article/p-mxrjjcnn-hn.htmlpython
Go語言內置的net/http
包十分的優秀,提供了HTTP客戶端和服務端的實現。json
Go語言內置的net/http
包提供了HTTP客戶端和服務端的實現。api
超文本傳輸協議(HTTP,HyperText Transfer Protocol)是互聯網上應用最爲普遍的一種網絡傳輸協議,全部的WWW文件都必須遵照這個標準。設計HTTP最初的目的是爲了提供一種發佈和接收HTML頁面的方法。瀏覽器
Get、Head、Post和PostForm函數發出HTTP/HTTPS請求。安全
resp, err := http.Get("http://example.com/") ... resp, err := http.Post("http://example.com/upload", "image/jpeg", &buf) ... resp, err := http.PostForm("http://example.com/form", url.Values{"key": {"Value"}, "id": {"123"}})
程序在使用完response後必須關閉回覆的主體。服務器
resp, err := http.Get("http://example.com/") if err != nil { // handle error } defer resp.Body.Close() body, err := ioutil.ReadAll(resp.Body) // ...
使用net/http
包編寫一個簡單的發送HTTP請求的Client端,代碼以下:網絡
package main import ( "fmt" "io/ioutil" "net/http" ) func main() { resp, err := http.Get("https://www.nickchen121.com/") if err != nil { fmt.Println("get failed, err:", err) return } defer resp.Body.Close() body, err := ioutil.ReadAll(resp.Body) if err != nil { fmt.Println("read from resp.Body failed,err:", err) return } fmt.Print(string(body)) }
將上面的代碼保存以後編譯成可執行文件,執行以後就能在終端打印nickchen121.com
網站首頁的內容了,咱們的瀏覽器其實就是一個發送和接收HTTP協議數據的客戶端,咱們平時經過瀏覽器訪問網頁其實就是從網站的服務器接收HTTP數據,而後瀏覽器會按照HTML、CSS等規則將網頁渲染展現出來。app
關於GET請求的參數須要使用Go語言內置的net/url
這個標準庫來處理。函數
func main() { apiUrl := "http://127.0.0.1:9090/get" // URL param data := url.Values{} data.Set("name", "小王子") data.Set("age", "18") u, err := url.ParseRequestURI(apiUrl) if err != nil { fmt.Printf("parse url requestUrl failed,err:%v\n", err) } u.RawQuery = data.Encode() // URL encode fmt.Println(u.String()) resp, err := http.Get(u.String()) if err != nil { fmt.Println("post failed, err:%v\n", err) return } defer resp.Body.Close() b, err := ioutil.ReadAll(resp.Body) if err != nil { fmt.Println("get resp failed,err:%v\n", err) return } fmt.Println(string(b)) }
對應的Server端HandlerFunc以下:
func getHandler(w http.ResponseWriter, r *http.Request) { defer r.Body.Close() data := r.URL.Query() fmt.Println(data.Get("name")) fmt.Println(data.Get("age")) answer := `{"status": "ok"}` w.Write([]byte(answer)) }
上面演示了使用net/http
包發送GET
請求的示例,發送POST
請求的示例代碼以下:
package main import ( "fmt" "io/ioutil" "net/http" "strings" ) // net/http post demo func main() { url := "http://127.0.0.1:9090/post" // 表單數據 //contentType := "application/x-www-form-urlencoded" //data := "name=小王子&age=18" // json contentType := "application/json" data := `{"name":"小王子","age":18}` resp, err := http.Post(url, contentType, strings.NewReader(data)) if err != nil { fmt.Println("post failed, err:%v\n", err) return } defer resp.Body.Close() b, err := ioutil.ReadAll(resp.Body) if err != nil { fmt.Println("get resp failed,err:%v\n", err) return } fmt.Println(string(b)) }
對應的Server端HandlerFunc以下:
func postHandler(w http.ResponseWriter, r *http.Request) { defer r.Body.Close() // 1. 請求類型是application/x-www-form-urlencoded時解析form數據 r.ParseForm() fmt.Println(r.PostForm) // 打印form數據 fmt.Println(r.PostForm.Get("name"), r.PostForm.Get("age")) // 2. 請求類型是application/json時從r.Body讀取數據 b, err := ioutil.ReadAll(r.Body) if err != nil { fmt.Println("read request.Body failed, err:%v\n", err) return } fmt.Println(string(b)) answer := `{"status": "ok"}` w.Write([]byte(answer)) }
要管理HTTP客戶端的頭域、重定向策略和其餘設置,建立一個Client:
client := &http.Client{ CheckRedirect: redirectPolicyFunc, } resp, err := client.Get("http://example.com") // ... req, err := http.NewRequest("GET", "http://example.com", nil) // ... req.Header.Add("If-None-Match", `W/"wyzzy"`) resp, err := client.Do(req) // ...
要管理代理、TLS配置、keep-alive、壓縮和其餘設置,建立一個Transport:
tr := &http.Transport{ TLSClientConfig: &tls.Config{RootCAs: pool}, DisableCompression: true, } client := &http.Client{Transport: tr} resp, err := client.Get("https://example.com")
Client和Transport類型均可以安全的被多個go程同時使用。出於效率考慮,應該一次創建、儘可能重用。
ListenAndServe使用指定的監聽地址和處理器啓動一個HTTP服務端。處理器參數一般是nil,這表示採用包變量DefaultServeMux做爲處理器。
Handle和HandleFunc函數能夠向DefaultServeMux添加處理器。
http.Handle("/foo", fooHandler) http.HandleFunc("/bar", func(w http.ResponseWriter, r *http.Request) { fmt.Fprintf(w, "Hello, %q", html.EscapeString(r.URL.Path)) }) log.Fatal(http.ListenAndServe(":8080", nil))
使用Go語言中的net/http
包來編寫一個簡單的接收HTTP請求的Server端示例,net/http
包是對net包的進一步封裝,專門用來處理HTTP協議的數據。具體的代碼以下:
// http server func sayHello(w http.ResponseWriter, r *http.Request) { fmt.Fprintln(w, "Hello 沙河!") } func main() { http.HandleFunc("/", sayHello) err := http.ListenAndServe(":9090", nil) if err != nil { fmt.Printf("http server failed, err:%v\n", err) return } }
將上面的代碼編譯以後執行,打開你電腦上的瀏覽器在地址欄輸入127.0.0.1:9090
回車,此時就可以看到以下頁面了。
要管理服務端的行爲,能夠建立一個自定義的Server:
s := &http.Server{ Addr: ":8080", Handler: myHandler, ReadTimeout: 10 * time.Second, WriteTimeout: 10 * time.Second, MaxHeaderBytes: 1 << 20, } log.Fatal(s.ListenAndServe())