写在前面
面试的时候问到如何关闭http请求,一般人脱口而出的是关闭response.body,这是错误的。response是返回结果的一个结构体,跟http连接没有关系。
type Response struct {
Status string // e.g. "200 OK"
StatusCode int // e.g. 200
Proto string // e.g. "HTTP/1.0"
ProtoMajor int // e.g. 1
ProtoMinor int // e.g. 0
Header Header
Body io.ReadCloser
ContentLength int64
Close bool
Uncompressed bool
Trailer Header
Request *Request
TLS *tls.ConnectionState
}
Body是Response中定义的一个IO流,用来读取返回内容。调用完成之后,无论http连接是否需要关闭,都要关闭response.body。
方式一:设置请求变量的 Close 字段值为 true
设置之后req.Close = true
,每次请求结束后就会主动关闭连接
func main() {
req, err := http.NewRequest("GET", "http://www.baidu.com", nil)
checkError(err)
req.Close = true
resp, err := http.DefaultClient.Do(req)
if resp != nil {
defer resp.Body.Close()
}
checkError(err)
body, err := ioutil.ReadAll(resp.Body)
checkError(err)
fmt.Println(string(body))
}
func checkError(err error) {
if err != nil {
fmt.Printf("err:%+v\n", err)
}
}
方式二:设置 Header 请求头部选项 Connection: close
设置req.Header.Add("Connection", "close")
之后,然后服务器返回的响应头部也会有这个选项,此时 HTTP 标准库会主动断开连接
func main() {
req, err := http.NewRequest("GET", "http://www.baidu.com", nil)
checkError(err)
req.Header.Add("Connection", "close")
resp, err := http.DefaultClient.Do(req)
if resp != nil {
defer resp.Body.Close()
}
checkError(err)
body, err := ioutil.ReadAll(resp.Body)
checkError(err)
fmt.Println(string(body))
}
func checkError(err error) {
if err != nil {
fmt.Printf("err:%+v\n", err)
}
}
方式三:自定义配置的 HTTP transport 客户端
这个主要是用来取消 HTTP 全局的复用连接,调用解释之后会自动关闭http连接
func main() {
tr := http.Transport{DisableKeepAlives: true}
client := http.Client{Transport: &tr}
resp, err := client.Get("http://www.baidu.com")
checkError(err)
if resp != nil {
defer resp.Body.Close()
}
body, err := ioutil.ReadAll(resp.Body)
checkError(err)
fmt.Println(string(body))
}
func checkError(err error) {
if err != nil {
fmt.Printf("err:%+v\n", err)
}
}
总结
每次学习一个知识点就记录一下,温故而知新,日积月累,终究会有蜕变!