golang內(nèi)置net/http包的使用詳解
簡(jiǎn)介
在當(dāng)今數(shù)字化的時(shí)代,Web應(yīng)用程序的開(kāi)發(fā)和網(wǎng)絡(luò)通信變得愈發(fā)重要。為了構(gòu)建強(qiáng)大的Web服務(wù)和客戶(hù)端,開(kāi)發(fā)者需要熟悉各種網(wǎng)絡(luò)協(xié)議和工具。而在Go語(yǔ)言中,內(nèi)置的net/http包提供了一種出色的方式來(lái)處理HTTP請(qǐng)求和響應(yīng),不僅功能強(qiáng)大,而且易于使用。本文將帶你深入了解Go語(yǔ)言?xún)?nèi)置的net/http包,揭示其強(qiáng)大的功能和用法。
Go語(yǔ)言?xún)?nèi)置的net/http包十分的優(yōu)秀,提供了HTTP客戶(hù)端和服務(wù)端的實(shí)現(xiàn)。
net/http介紹 Go語(yǔ)言?xún)?nèi)置的net/http包提供了HTTP客戶(hù)端和服務(wù)端的實(shí)現(xiàn)。
HTTP協(xié)議
超文本傳輸協(xié)議(HTTP,HyperText Transfer Protocol)是互聯(lián)網(wǎng)上應(yīng)用最為廣泛的一種網(wǎng)絡(luò)傳輸協(xié)議,所有的WWW文件都必須遵守這個(gè)標(biāo)準(zhǔn)。設(shè)計(jì)HTTP最初的目的是為了提供一種發(fā)布和接收HTML頁(yè)面的方法。
HTTP客戶(hù)端
基本的HTTP/HTTPS請(qǐng)求 Get、Head、Post和PostForm函數(shù)發(fā)出HTTP/HTTPS請(qǐng)求。
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請(qǐng)求示例
使用net/http包編寫(xiě)一個(gè)簡(jiǎn)單的發(fā)送HTTP請(qǐng)求的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)站首頁(yè)的內(nèi)容了,我們的瀏覽器其實(shí)就是一個(gè)發(fā)送和接收HTTP協(xié)議數(shù)據(jù)的客戶(hù)端,我們平時(shí)通過(guò)瀏覽器訪問(wèn)網(wǎng)頁(yè)其實(shí)就是從網(wǎng)站的服務(wù)器接收HTTP數(shù)據(jù),然后瀏覽器會(huì)按照HTML、CSS等規(guī)則將網(wǎng)頁(yè)渲染展示出來(lái)。
帶參數(shù)的GET請(qǐng)求示例
關(guān)于GET請(qǐng)求的參數(shù)需要使用Go語(yǔ)言?xún)?nèi)置的net/url這個(gè)標(biāo)準(zhǔn)庫(kù)來(lái)處理。
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))
}
對(duì)應(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請(qǐng)求示例
上面演示了使用net/http包發(fā)送GET請(qǐng)求的示例,發(fā)送POST請(qǐng)求的示例代碼如下:
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))
}
對(duì)應(yīng)的Server端HandlerFunc如下:
func postHandler(w http.ResponseWriter, r *http.Request) {
defer r.Body.Close()
// 1. 請(qǐng)求類(lèi)型是application/x-www-form-urlencoded時(shí)解析form數(shù)據(jù)
r.ParseForm()
fmt.Println(r.PostForm) // 打印form數(shù)據(jù)
fmt.Println(r.PostForm.Get("name"), r.PostForm.Get("age"))
// 2. 請(qǐng)求類(lèi)型是application/json時(shí)從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客戶(hù)端的頭域、重定向策略和其他設(shè)置,創(chuàng)建一個(gè)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)建一個(gè)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類(lèi)型都可以安全的被多個(gè)goroutine同時(shí)使用。出于效率考慮,應(yīng)該一次建立、盡量重用。
服務(wù)端
默認(rèn)的Server
ListenAndServe使用指定的監(jiān)聽(tīng)地址和處理器啟動(dòng)一個(gè)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語(yǔ)言中的net/http包來(lái)編寫(xiě)一個(gè)簡(jiǎn)單的接收HTTP請(qǐng)求的Server端示例,net/http包是對(duì)net包的進(jìn)一步封裝,專(zhuān)門(mén)用來(lái)處理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í)行,打開(kāi)你電腦上的瀏覽器在地址欄輸入127.0.0.1:9090回車(chē),此時(shí)就能夠看到。hello頁(yè)面
自定義Server 要管理服務(wù)端的行為,可以創(chuàng)建一個(gè)自定義的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語(yǔ)言中,net/http包是構(gòu)建高性能Web應(yīng)用程序的強(qiáng)大工具,它提供了豐富的功能和簡(jiǎn)單的API,使開(kāi)發(fā)者能夠輕松創(chuàng)建HTTP客戶(hù)端和服務(wù)端。通過(guò)本文的示例和介紹,你已經(jīng)了解了如何發(fā)送GET和POST請(qǐng)求,處理參數(shù),自定義客戶(hù)端和服務(wù)端,以及創(chuàng)建自定義路由處理器。這將為您的Go語(yǔ)言開(kāi)發(fā)工作提供堅(jiān)實(shí)的基礎(chǔ),讓您能夠構(gòu)建出色的Web應(yīng)用程序和服務(wù)。繼續(xù)探索net/http包的強(qiáng)大功能,并在你的下一個(gè)項(xiàng)目中充分利用它。
以上就是golang內(nèi)置net/http包的使用詳解的詳細(xì)內(nèi)容,更多關(guān)于go內(nèi)置http包的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
- 詳解Go語(yǔ)言中net/http包的使用
- go標(biāo)準(zhǔn)庫(kù)net/http服務(wù)端的實(shí)現(xiàn)示例
- 一文帶你吃透Golang中net/http標(biāo)準(zhǔn)庫(kù)服務(wù)端
- Go語(yǔ)言使用net/http實(shí)現(xiàn)簡(jiǎn)單登錄驗(yàn)證和文件上傳功能
- golang中的net/http庫(kù)基本使用詳解
- 一文詳解Golang中net/http包的實(shí)現(xiàn)原理
- Go 1.22對(duì)net/http包的路由增強(qiáng)功能詳解
- Go net/http的簡(jiǎn)單使用小結(jié)
- Go語(yǔ)言net/http庫(kù)使用詳解
相關(guān)文章
Golang使用gin框架實(shí)現(xiàn)一個(gè)完整的聊天室功能
由于我們項(xiàng)目的需要,我就研究了一下關(guān)于websocket的相關(guān)內(nèi)容,去實(shí)現(xiàn)一個(gè)聊天室的功能,經(jīng)過(guò)幾天的探索,現(xiàn)在使用Gin框架實(shí)現(xiàn)了一個(gè)完整的聊天室+消息實(shí)時(shí)通知系統(tǒng),感興趣的小伙伴歡迎閱讀本文2023-08-08
Go語(yǔ)言kube-scheduler深度剖析開(kāi)發(fā)之scheduler初始化
這篇文章主要介紹了Go語(yǔ)言kube-scheduler深度剖析開(kāi)發(fā)之scheduler初始化實(shí)現(xiàn)過(guò)程示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-04-04
一文搞懂Golang 時(shí)間和日期相關(guān)函數(shù)
這篇文章主要介紹了Golang 時(shí)間和日期相關(guān)函數(shù),本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-12-12
golang bad file descriptor問(wèn)題的解決方法
這篇文章主要給大家介紹了golang bad file descriptor問(wèn)題的解決方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-02-02

