1. 程式人生 > >golang中傳送http請求的幾種常見情況

golang中傳送http請求的幾種常見情況

整理一下golang中各種http的傳送方式

最初的一些專案中,很多的地方都使用到了golang傳送http請求,之後再對請求收到的結果進行一些處理,所用到的模式也比較固定,這裡整理一下集中http的傳送方式,先記錄這麼多,再一點一點新增。

最基本的場景

方式一 使用http.Newrequest

先生成http.client -> 再生成 http.request -> 之後提交請求:client.Do(request) -> 處理返回結果,每一步的過程都可以設定一些具體的引數,下面是一個最樸素最基本的例子:


//question ???將stdout重定向為response資訊???
package main

import (
	"fmt"
	"io"
	"net/http"
	"os"
)

func main() {
    //生成client 引數為預設
	client := &http.Client{}
	
	//生成要訪問的url
	url := "http://www.baidu.com"
	    
	//提交請求
	reqest, err := http.NewRequest("GET", url, nil)
	
	if err != nil {
		panic(err)
	}
	
	//處理返回結果
	response, _ := client.Do(reqest)
   
   //將結果定位到標準輸出 也可以直接打印出來 或者定位到其他地方進行相應的處理
	stdout := os.Stdout
	_, err = io.Copy(stdout, response.Body)
   
   //返回的狀態碼
	status := response.StatusCode

	fmt.Println(status)

}

方式二 先生成client,之後用client.get/post..

client結構自己也有一些傳送api的方法,比如client.get,client.post,client.postform..等等。基本上涵蓋了主要的http請求的型別,通常不進行什麼特殊的配置的話,這樣就可以了,其實client的get或者post方法,也是對http.Newerequest方法的封裝,裡面還額外添加了req.Header.Set("Content-Type", bodyType)一般用的話,也是ok的

方式三 http. Get/Post..

具體實現的時候,還是採用的先前提到的模式,先生成一個預設的client,之後呼叫http.Newrequest方法。

對每個步驟進行細節性的說明

生成client時候的引數配置

最常見的一個引數是使用https的方式傳送資訊時候client端的設定。如果生成client的時候,什麼資訊都不新增,就會使用預設的值。具體的資訊包括:

	Transport RoundTripper

	CheckRedirect func(req *Request, via []*Request) error

	Jar CookieJar

	Timeout time.Duration

第一個引數是一個RoundTripper介面,裡面包含了一個RoundTrip函式,指定了一些http請求的基本機制。http.Transport中涉及到的引數較多,要是不指定的話,就會使用預設的DefaultTransport引數,裡面包含一些預設的請求時間以及proxy機制之類的。具體的細節引數涉及到好多,有的都沒有使用到過比如那些我握手時間之類的,目前使用到的最多的就是https的相關引數:TLSClientConfig,這是一個*tls.Config型別,其中涉及到的引數還是有很多,一個基本的是用案例如下,僅僅是在配置中制定了rooca以及客戶度端使用的證書。相關的https的內容可以參考之前的

這一篇

通常傳送https請求的時候,前面的引數可以使用如下方式進行處理:

    pool := x509.NewCertPool()
	caCertPath := "certs/cert_server/ca.crt"

	caCrt, err := ioutil.ReadFile(caCertPath)
	if err != nil {
		fmt.Println("ReadFile err:", err)
		return
	}
	pool.AppendCertsFromPEM(caCrt)

	cliCrt, err := tls.LoadX509KeyPair("certs/cert_server/client.crt", "certs/cert_server/client.key")
	if err != nil {
		fmt.Println("Loadx509keypair err:", err)
		return
	}
	
	tr := &http.Transport{
		TLSClientConfig: &tls.Config{
			RootCAs:      pool,
			Certificates: []tls.Certificate{cliCrt},
		},
	}
	client := &http.Client{Transport: tr}

生成request時候的引數配置

生成request的時候,主要的是幾個基本的引數。NewRequest函式有三個基本的引數,NewRequest(method, urlStr string, body io.Reader)第一個是請求的型別,GET, POST, PUT, etc.要設成大寫的形式。第二個引數是請求要訪問的url,第三個引數是請求的body中的內容,需要是一個io.Reader的型別。

注意io.Reader的介面中是一個Read方法,實現了Read方法的型別應該都可以作為io.Reader來返回,Read(p []byte) (n int, err error)函式具體的功能就是讀入len(p)長度的內容到p中,返回讀入的長度以及錯誤資訊。

通常是採用strings.NewReader函式,將一個string型別轉化為io.Reader型別,或者bytes.NewBuffer函式,將[]byte型別轉化為io.Reader型別。

此外還可以給request的header中新增一些額外的資訊,比如下面例子中添加了請求的body的型別以及token的資訊。

	reqest.Header.Set("Content-Type", "application/x-www-form-urlencoded")
	reqest.Header.Set("Authorization", "qwertyuiopasdfghjklzxcvbnm1234567890")

還有比如模擬表單提交,可以把提交的型別設定為url.Values型別再進行Encode:

// use map as struct
	var clusterinfo = url.Values{}
	//var clusterinfo = map[string]string{}
	clusterinfo.Add("userName", user)
	clusterinfo.Add("password", pw)
	clusterinfo.Add("cloudName", clustername)
	clusterinfo.Add("masterIp", masterip)
	clusterinfo.Add("cacrt", string(caCrt))

	data := clusterinfo.Encode()
	
	url := "https://10.10.105.124:8443/user/checkAndUpdate"
	reqest, err := http.NewRequest("POST", url, strings.NewReader(data))

最常見的一種情況是傳送一個json檔案過去,可以把Header的型別設定成為:

"Content-Type", "application/json; charset=utf-8"

其餘的部分按照先前同樣的方式進行設定傳送提交就好。

request的型別的屬性還是比較多的,慢慢整理。

生成的response結果的處理

一般在client構建好之後,要採用client.Do(request)方法提交client請求,之後會返回一個*Response型別。response中的引數一般也比較多,我們需要的最多的通常是Body引數,一般通過body, _ := ioutil.ReadAll(resp.Body)會把body轉化為[]byte型別返回過來, 之後再進行其他的處理。