golang关于cookiejar用法的疑问

看文章上写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

阅读 14.2k
2 个回答

net/http/client.go#L167

// didTimeout is non-nil only if err != nil.
func (c *Client) send(req *Request, deadline time.Time) (resp *Response, didTimeout func() bool, err error) {
    if c.Jar != nil {
        for _, cookie := range c.Jar.Cookies(req.URL) {
            req.AddCookie(cookie)  // 这里
        }
    }
    resp, didTimeout, err = send(req, c.transport(), deadline)
    if err != nil {
        return nil, didTimeout, err
    }
    if c.Jar != nil {
        if rc := resp.Cookies(); len(rc) > 0 {
            c.Jar.SetCookies(req.URL, rc) // 这里
        }
    }
    return resp, nil, nil
}

最根本的原因是多次client.Do()用了相同的req.

新手上路,请多包涵

复用了一个请求

推荐问题
宣传栏