看文章上写cookiejar会自动帮忙管理服务端返回的cookies,于是我做了个实验。
实现服务端
package main
import (
"fmt"
"net/http"
"strconv"
"time"
)
var i int
func receiveReq(w http.ResponseWriter, r *http.Request) {
//i记录请求的次数
i++
fmt.Printf("第 %d 次请求的Cookie: ", i)
fmt.Println(r.Cookies())
//设置cookie
tNow := time.Now()
cookie := &http.Cookie{
Name: "XMEN",
Value: "STORM" + strconv.Itoa(i),
Expires: tNow.AddDate(1, 0, i),
}
http.SetCookie(w, cookie)
//返回信息
w.Write(([]byte("your cookie has been received")))
}
func main() {
fmt.Println("Server Start Now!")
http.HandleFunc("/test", receiveReq)
err := http.ListenAndServe("127.0.0.1:8889", nil)
if err != nil {
fmt.Println("ListenAndServe ERROR: ", err)
}
}
实现client端
package main
import (
"fmt"
"net/http"
"net/http/cookiejar"
)
func main() {
jar, _ := cookiejar.New(nil)
fmt.Println("Start Request Server")
client := http.Client{
Jar: jar,
}
url := "http://127.0.0.1:8889/test"
req, _ := http.NewRequest("GET", url, nil)
//第一次发请求
client.Do(req)
fmt.Printf("第一次 %s \n", req.Cookies())
//第二次发请求
client.Do(req)
fmt.Printf("第二次 %s \n", req.Cookies())
//第三次发请求
client.Do(req)
fmt.Printf("第三次 %s \n", req.Cookies())
//第四次发请求
client.Do(req)
fmt.Printf("第四次 %s \n", req.Cookies())
//第五次发请求
client.Do(req)
fmt.Printf("第五次 %s \n", req.Cookies())
}
先通过浏览器请求5次,再通过client端请求5次
Server Start Now!
第 1 次请求的Cookie: []
第 2 次请求的Cookie: [XMEN=STORM1]
第 3 次请求的Cookie: [XMEN=STORM2]
第 4 次请求的Cookie: [XMEN=STORM3]
第 5 次请求的Cookie: [XMEN=STORM4]
第 6 次请求的Cookie: []
第 7 次请求的Cookie: [XMEN=STORM6]
第 8 次请求的Cookie: [XMEN=STORM6 XMEN=STORM7]
第 9 次请求的Cookie: [XMEN=STORM6 XMEN=STORM7 XMEN=STORM8]
第 10 次请求的Cookie: [XMEN=STORM6 XMEN=STORM7 XMEN=STORM8 XMEN=STORM9]
那么问题来了:cookiejar为什么会存留所有历史cookie
net/http/client.go#L167
最根本的原因是多次client.Do()用了相同的req.