golang內置net/http包的使用詳解
簡介
在當今數(shù)字化的時代,Web應用程序的開發(fā)和網絡通信變得愈發(fā)重要。為了構建強大的Web服務和客戶端,開發(fā)者需要熟悉各種網絡協(xié)議和工具。而在Go語言中,內置的net/http包提供了一種出色的方式來處理HTTP請求和響應,不僅功能強大,而且易于使用。本文將帶你深入了解Go語言內置的net/http包,揭示其強大的功能和用法。
Go語言內置的net/http包十分的優(yōu)秀,提供了HTTP客戶端和服務端的實現(xiàn)。
net/http介紹 Go語言內置的net/http包提供了HTTP客戶端和服務端的實現(xiàn)。
HTTP協(xié)議
超文本傳輸協(xié)議(HTTP,HyperText Transfer Protocol)是互聯(lián)網上應用最為廣泛的一種網絡傳輸協(xié)議,所有的WWW文件都必須遵守這個標準。設計HTTP最初的目的是為了提供一種發(fā)布和接收HTML頁面的方法。
HTTP客戶端
基本的HTTP/HTTPS請求 Get、Head、Post和PostForm函數(shù)發(fā)出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) // ...
GET請求示例
使用net/http包編寫一個簡單的發(fā)送HTTP請求的Client端,代碼如下:
package main ? import ( "fmt" "io/ioutil" "net/http" ) ? func main() { resp, err := http.Get("https://www.liwenzhou.com/") if err != nil { fmt.Printf("get failed, err:%v\n", err) return } defer resp.Body.Close() body, err := ioutil.ReadAll(resp.Body) if err != nil { fmt.Printf("read from resp.Body failed, err:%v\n", err) return } fmt.Print(string(body)) }
將上面的代碼保存之后編譯成可執(zhí)行文件,執(zhí)行之后就能在終端打印liwenzhou.com網站首頁的內容了,我們的瀏覽器其實就是一個發(fā)送和接收HTTP協(xié)議數(shù)據(jù)的客戶端,我們平時通過瀏覽器訪問網頁其實就是從網站的服務器接收HTTP數(shù)據(jù),然后瀏覽器會按照HTML、CSS等規(guī)則將網頁渲染展示出來。
帶參數(shù)的GET請求示例
關于GET請求的參數(shù)需要使用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.Printf("post failed, err:%v\n", err) return } defer resp.Body.Close() b, err := ioutil.ReadAll(resp.Body) if err != nil { fmt.Printf("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)) }
Post請求示例
上面演示了使用net/http包發(fā)送GET請求的示例,發(fā)送POST請求的示例代碼如下:
package main ? import ( "fmt" "io/ioutil" "net/http" "strings" ) ? // net/http post demo ? func main() { url := "http://127.0.0.1:9090/post" // 表單數(shù)據(jù) //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.Printf("post failed, err:%v\n", err) return } defer resp.Body.Close() b, err := ioutil.ReadAll(resp.Body) if err != nil { fmt.Printf("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數(shù)據(jù) r.ParseForm() fmt.Println(r.PostForm) // 打印form數(shù)據(jù) fmt.Println(r.PostForm.Get("name"), r.PostForm.Get("age")) // 2. 請求類型是application/json時從r.Body讀取數(shù)據(jù) b, err := ioutil.ReadAll(r.Body) if err != nil { fmt.Printf("read request.Body failed, err:%v\n", err) return } fmt.Println(string(b)) answer := `{"status": "ok"}` w.Write([]byte(answer)) }
自定義Client
要管理HTTP客戶端的頭域、重定向策略和其他設置,創(chuàng)建一個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) // ...
自定義Transport
要管理代理、TLS配置、keep-alive、壓縮和其他設置,創(chuàng)建一個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類型都可以安全的被多個goroutine同時使用。出于效率考慮,應該一次建立、盡量重用。
服務端
默認的Server
ListenAndServe使用指定的監(jiān)聽地址和處理器啟動一個HTTP服務端。處理器參數(shù)通常是nil,這表示采用包變量DefaultServeMux作為處理器。
Handle和HandleFunc函數(shù)可以向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))
默認的Server示例
使用Go語言中的net/http包來編寫一個簡單的接收HTTP請求的Server端示例,net/http包是對net包的進一步封裝,專門用來處理HTTP協(xié)議的數(shù)據(jù)。具體的代碼如下:
// 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 } }
將上面的代碼編譯之后執(zhí)行,打開你電腦上的瀏覽器在地址欄輸入127.0.0.1:9090
回車,此時就能夠看到。hello頁面
自定義Server 要管理服務端的行為,可以創(chuàng)建一個自定義的Server:
s := &http.Server{ Addr: ":8080", Handler: myHandler, ReadTimeout: 10 * time.Second, WriteTimeout: 10 * time.Second, MaxHeaderBytes: 1 << 20, } log.Fatal(s.ListenAndServe())
總結
在Go語言中,net/http包是構建高性能Web應用程序的強大工具,它提供了豐富的功能和簡單的API,使開發(fā)者能夠輕松創(chuàng)建HTTP客戶端和服務端。通過本文的示例和介紹,你已經了解了如何發(fā)送GET和POST請求,處理參數(shù),自定義客戶端和服務端,以及創(chuàng)建自定義路由處理器。這將為您的Go語言開發(fā)工作提供堅實的基礎,讓您能夠構建出色的Web應用程序和服務。繼續(xù)探索net/http包的強大功能,并在你的下一個項目中充分利用它。
以上就是golang內置net/http包的使用詳解的詳細內容,更多關于go內置http包的資料請關注腳本之家其它相關文章!
相關文章
go?gin?正確讀取http?response?body內容并多次使用詳解
這篇文章主要為大家介紹了go?gin?正確讀取http?response?body內容并多次使用解決思路,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-01-01Go語言基于viper實現(xiàn)apollo多實例快速
viper是適用于go應用程序的配置解決方案,這款配置管理神器,支持多種類型、開箱即用、極易上手。本文主要介紹了如何基于viper實現(xiàn)apollo多實例快速接入,感興趣的可以了解一下2023-01-01