1. 程式人生 > 其它 >Go基礎庫之net/http

Go基礎庫之net/http

Go語言內建的net/http包十分的優秀,提供了HTTP客戶端和服務端的實現。

net/http介紹

Go語言內建的net/http包提供了HTTP客戶端和服務端的實現。

HTTP協議

超文字傳輸協議(HTTP,HyperText Transfer Protocol)是網際網路上應用最為廣泛的一種網路傳輸協議,所有的WWW檔案都必須遵守這個標準。設計HTTP最初的目的是為了提供一種釋出和接收HTML頁面的方法。

HTTP客戶端

基本的HTTP/HTTPS請求

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) // ...

 

 

[GET]

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)) }

帶引數:

引數需要使用到net/url來處理

func main() {
    apiUrl := "http://127.0.0.1:9090/get"
    // URL param
    data := url.Values{}
    data.Set("name", "mi")
    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端:

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]

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=mi&age=18"
    // json
    contentType := "application/json"
    data := `{"name":"mi","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端:

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.Printf("read request.Body failed, err:%v\n", err)
        return
    }
    fmt.Println(string(b))
    answer := `{"status": "ok"}`
    w.Write([]byte(answer))
}

 

自定義Client

要管理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)

自定義Transport

要管理代理、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型別都可以安全的被多個goroutine同時使用。出於效率考慮,應該一次建立、儘量重用。

 

 

服務端

預設的Server

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))

預設的Server示例

使用Go語言中的net/http包來編寫一個簡單的接收HTTP請求的Server端示例,net/http包是對net包的進一步封裝,專門用來處理HTTP協議的資料。具體的程式碼如下:

package main

import (
    "fmt"
    "net/http"
)

func sayHello(w http.ResponseWriter, r *http.Request) {
    fmt.Fprintln(w, "Hello shanghai!")
}

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

要管理服務端的行為,可以建立一個自定義的Server:

s := &http.Server{
    Addr:           ":8080",
    Handler:        myHandler,
    ReadTimeout:    10 * time.Second,
    WriteTimeout:   10 * time.Second,
    MaxHeaderBytes: 1 << 20,
}
log.Fatal(s.ListenAndServe())

 

 

 demo:

package main

import (
    "encoding/json"
    _"encoding/json"
    "fmt"
    "io/ioutil"
    "net/http"
    "net/url"
    "strings"
)

var (
    api = "http://xxxxx.com"
    token = "9adf92861"
)

type Result struct {
    Code int `json:"code"`
    Msg string `json:"msg"`
}

type Resp struct {
    Code string `json:"code"`
    Msg string `json:"msg"`
}


func wechatUrlHandle(w http.ResponseWriter, r *http.Request){
    // 獲取連結地址
    var wechatUrl string
    values := r.URL.Query()
    wechatUrl = values.Get("url")
    // 驗證URL的正確性
    targetUrl, err := url.ParseRequestURI(wechatUrl)
    if err != nil {
        fmt.Printf("parse url failed, err:%v\n", err)
    }
    fmt.Printf(fmt.Sprintf("url: %v", targetUrl))

    // 拼接引數發起請求
    data := fmt.Sprintf("token=%s&url=%s", token, targetUrl.String())

    headers := "application/x-www-form-urlencoded"
    res, err := http.Post(api, headers, strings.NewReader(data))
    if err != nil {
        fmt.Printf("post failed, err: %v\n", err)
        return
    }

    // 關閉請求資源
    defer res.Body.Close()
    body, err := ioutil.ReadAll(res.Body)
    if err != nil{
        fmt.Fprintf(w, fmt.Sprintf("get resp failed. err: %v\n", err))
        return
    }
    
    var info Result
    var ress Resp
    w.Header().Set("Content-Type", "application/json")
    if err := json.Unmarshal(body, &info); err == nil {
        fmt.Println(info.Code, info.Msg)
        ress.Code = "200"
        ress.Msg = "Success"
        t, _ := json.Marshal(ress)
        w.Write(t)
    }else{
        ress.Code = "200"
        ress.Msg = "failure"
        t, _ := json.Marshal(ress)
        fmt.Println("err: ", err)
        w.Write(t)
    }
}

func main() {
    
    http.HandleFunc("/wechat/url", wechatUrlHandle)
    err := http.ListenAndServe(":9090", nil)  // 處理器引數一般未nil
    if err != nil {
        fmt.Printf("Http server failed. err: %v\n", err)
        return
    }

}