golang內(nèi)置net/http包的使用詳解
簡介
在當(dāng)今數(shù)字化的時代,Web應(yīng)用程序的開發(fā)和網(wǎng)絡(luò)通信變得愈發(fā)重要。為了構(gòu)建強(qiáng)大的Web服務(wù)和客戶端,開發(fā)者需要熟悉各種網(wǎng)絡(luò)協(xié)議和工具。而在Go語言中,內(nèi)置的net/http包提供了一種出色的方式來處理HTTP請求和響應(yīng),不僅功能強(qiáng)大,而且易于使用。本文將帶你深入了解Go語言內(nèi)置的net/http包,揭示其強(qiáng)大的功能和用法。
Go語言內(nèi)置的net/http包十分的優(yōu)秀,提供了HTTP客戶端和服務(wù)端的實(shí)現(xiàn)。
net/http介紹 Go語言內(nèi)置的net/http包提供了HTTP客戶端和服務(wù)端的實(shí)現(xiàn)。
HTTP協(xié)議
超文本傳輸協(xié)議(HTTP,HyperText Transfer Protocol)是互聯(lián)網(wǎng)上應(yīng)用最為廣泛的一種網(wǎng)絡(luò)傳輸協(xié)議,所有的WWW文件都必須遵守這個標(biāo)準(zhǔn)。設(shè)計(jì)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后必須關(guān)閉回復(fù)的主體。
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網(wǎng)站首頁的內(nèi)容了,我們的瀏覽器其實(shí)就是一個發(fā)送和接收HTTP協(xié)議數(shù)據(jù)的客戶端,我們平時通過瀏覽器訪問網(wǎng)頁其實(shí)就是從網(wǎng)站的服務(wù)器接收HTTP數(shù)據(jù),然后瀏覽器會按照HTML、CSS等規(guī)則將網(wǎng)頁渲染展示出來。
帶參數(shù)的GET請求示例
關(guān)于GET請求的參數(shù)需要使用Go語言內(nèi)置的net/url這個標(biāo)準(zhǔn)庫來處理。
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)) }
對應(yīng)的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)) }
對應(yīng)的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客戶端的頭域、重定向策略和其他設(shè)置,創(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、壓縮和其他設(shè)置,創(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同時使用。出于效率考慮,應(yīng)該一次建立、盡量重用。
服務(wù)端
默認(rèn)的Server
ListenAndServe使用指定的監(jiān)聽地址和處理器啟動一個HTTP服務(wù)端。處理器參數(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))
默認(rèn)的Server示例
使用Go語言中的net/http包來編寫一個簡單的接收HTTP請求的Server端示例,net/http包是對net包的進(jìn)一步封裝,專門用來處理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 要管理服務(wù)端的行為,可以創(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())
總結(jié)
在Go語言中,net/http包是構(gòu)建高性能Web應(yīng)用程序的強(qiáng)大工具,它提供了豐富的功能和簡單的API,使開發(fā)者能夠輕松創(chuàng)建HTTP客戶端和服務(wù)端。通過本文的示例和介紹,你已經(jīng)了解了如何發(fā)送GET和POST請求,處理參數(shù),自定義客戶端和服務(wù)端,以及創(chuàng)建自定義路由處理器。這將為您的Go語言開發(fā)工作提供堅(jiān)實(shí)的基礎(chǔ),讓您能夠構(gòu)建出色的Web應(yīng)用程序和服務(wù)。繼續(xù)探索net/http包的強(qiáng)大功能,并在你的下一個項(xiàng)目中充分利用它。
以上就是golang內(nèi)置net/http包的使用詳解的詳細(xì)內(nèi)容,更多關(guān)于go內(nèi)置http包的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
如何利用Go語言實(shí)現(xiàn)LRU?Cache
這篇文章主要介紹了如何利用Go語言實(shí)現(xiàn)LRU?Cache,LRU是Least?Recently?Used的縮寫,是一種操作系統(tǒng)中常用的頁面置換算法,下面我們一起進(jìn)入文章了解更多內(nèi)容吧,需要的朋友可以參考一下2022-03-03golang進(jìn)程在docker中OOM后hang住問題解析
這篇文章主要介紹了golang進(jìn)程在docker中OOM后hang住問題解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-10-10Go Gin框架中的binding驗(yàn)證器使用小結(jié)
Gin框架中的binding驗(yàn)證器為我們提供了簡便的數(shù)據(jù)綁定和驗(yàn)證功能,通過合理使用binding和validate標(biāo)簽,我們可以確保API接口的數(shù)據(jù)合法性和完整性,這篇文章主要介紹了Go Gin框架中的binding驗(yàn)證器使用指南,需要的朋友可以參考下2024-07-07Golang科學(xué)計(jì)數(shù)法轉(zhuǎn)換string數(shù)字輸出的實(shí)現(xiàn)
最近接手一個商城運(yùn)單號模塊,接手后發(fā)現(xiàn)有部分運(yùn)單號返回給前端是按照科學(xué)計(jì)數(shù)法的方式返回,本文就介紹一下Golang科學(xué)計(jì)數(shù)法轉(zhuǎn)換string數(shù)字輸出,感興趣的可以了解一下2021-07-07