解決golang http重定向失效的問題
最近在學習GoLang,在使用http重定向的時候發現了一個很有趣的現象,在這裡記錄一下。
r.GET("/index",func(c *gin.Context) { c.Redirect(http.StatusMovedPermanently,"http://www.baidu.com/") })
本來寫了這麼一段程式碼,將我的路由重定向到“百度”,第一次試驗成功了。之後當我想重新定向到其它網站,或者不重定向而試驗其它邏輯的時候,發現在瀏覽器中,永遠只是定向到“百度”。非常疑惑,明明程式都重新運行了呀。後來我發現,用postman或者另一個瀏覽器開啟,我新寫的邏輯是能實現的。
最後清空了瀏覽器近一小時的記錄,發現能實現新邏輯了。
所以應該是因為瀏覽器快取導致直接在快取中取內容,而不是從我服務端。
補充:golang不想http自動處理重定向的解決方案
前言
有時候傳送http請求不想讓庫自動幫忙處理重定向,庫裡面預設的是會把所有重定向都完成一遍,結果就是最後一個沒有重定向的請求的結果。
因此需要一種方案直接獲取首次訪問的結果,不走重定向。
go的http庫裡面是使用如下程式碼檢查重定向的,以前我傻傻的修改原始碼讓下面這段程式碼直接返回,這樣需要重新編譯go自帶的庫,後來發現更簡單的方案。
if err == ErrUseLastResponse { return resp,nil // 這裡是攔截重定向,如果不攔截則走下面的重定向判斷 } var shouldRedirect bool redirectMethod,shouldRedirect,includeBody = redirectBehavior(req.Method,resp,reqs[0]) if !shouldRedirect { return resp,nil }
解決方案
下面程式碼可以驗證自動處理重定向,以及不走重定向的方案。
package main import ( "io/ioutil" "log" "net/http" "time" ) func main() { go server() time.Sleep(time.Second) mUrl := "http://127.0.0.1:12345/post" { // 常規方法 req,err := http.NewRequest(http.MethodPost,mUrl,nil) if err != nil { log.Fatal(err) } resp,err := http.DefaultClient.Do(req) if resp != nil { defer resp.Body.Close() } if err != nil { log.Fatal(err) } byt,err := ioutil.ReadAll(resp.Body) if err != nil { log.Fatal(err) } log.Println(resp.StatusCode,"|",string(byt[:128])) } { // 去掉自動處理重定向 req,err := http.DefaultTransport.RoundTrip(req) if resp != nil { defer resp.Body.Close() } if err != nil { log.Fatal(err) } byt,string(byt[:128])) } { // 另一種不要重定向的方法 req,nil) if err != nil { log.Fatal(err) } client := &http.Client{ CheckRedirect: func(req *http.Request,via []*http.Request) error { return http.ErrUseLastResponse /* 不進入重定向 */ },} resp,err := client.Do(req) if resp != nil { defer resp.Body.Close() } if err != nil { log.Fatal(err) } byt,string(byt[:128])) } } // 下面開啟一個服務,重定向到百度 func server() { http.HandleFunc("/post",mPost) http.ListenAndServe(":12345",nil) } func mPost(w http.ResponseWriter,r *http.Request) { http.Redirect(w,r,"http://www.baidu.com",http.StatusFound) w.Write([]byte(time.Now().String())) }
結論
如下的第一個請求是直接返回百度的網頁,及重定向以後的內容。第二個請求直接返回第一個302重定向的內容。
2020/10/14 13:11:56 200 | 百度一下,你就知道
2020/10/14 13:11:56 302 | 2020-10-14 13:11:56.6559382 +0800 CST m=+1.429170501
2020/10/14 13:11:56 302 | 2020-10-14 13:11:56.6559382 +0800 CST m=+1.429170501
以上為個人經驗,希望能給大家一個參考,也希望大家多多支援我們。如有錯誤或未考慮完全的地方,望不吝賜教。