解決golang http重定向失效的問題
最近在學習GoLang,在使用http重定向的時候發(fā)現(xiàn)了一個很有趣的現(xiàn)象,在這里記錄一下。
r.GET("/index", func(c *gin.Context) { c.Redirect(http.StatusMovedPermanently, "http://www.baidu.com/") })
本來寫了這么一段代碼,將我的路由重定向到“百度”,第一次試驗成功了。之后當我想重新定向到其它網站,或者不重定向而試驗其它邏輯的時候,發(fā)現(xiàn)在瀏覽器中,永遠只是定向到“百度”。非常疑惑,明明程序都重新運行了呀。后來我發(fā)現(xiàn),用postman或者另一個瀏覽器打開,我新寫的邏輯是能實現(xiàn)的。
最后清空了瀏覽器近一小時的記錄,發(fā)現(xiàn)能實現(xiàn)新邏輯了。
所以應該是因為瀏覽器緩存導致直接在緩存中取內容,而不是從我服務端。
補充:golang不想http自動處理重定向的解決方案
前言
有時候發(fā)送http請求不想讓庫自動幫忙處理重定向,庫里面默認的是會把所有重定向都完成一遍,結果就是最后一個沒有重定向的請求的結果。
因此需要一種方案直接獲取首次訪問的結果,不走重定向。
go的http庫里面是使用如下代碼檢查重定向的,以前我傻傻的修改源碼讓下面這段代碼直接返回,這樣需要重新編譯go自帶的庫,后來發(fā)現(xiàn)更簡單的方案。
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" { // 常規(guī)方法 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.NewRequest(http.MethodPost, mUrl, nil) if err != nil { log.Fatal(err) } resp, err := http.DefaultTransport.RoundTrip(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.NewRequest(http.MethodPost, mUrl, 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, err := ioutil.ReadAll(resp.Body) if err != nil { log.Fatal(err) } log.Println(resp.StatusCode, "|", 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
以上為個人經驗,希望能給大家一個參考,也希望大家多多支持腳本之家。如有錯誤或未考慮完全的地方,望不吝賜教。
相關文章
搭建Go語言的ORM框架Gorm的具體步驟(從Java到go)
很多朋友不知道如何使用Goland軟件,搭建一個ORM框架GORM,今天小編給大家分享一篇教程關于搭建Go語言的ORM框架Gorm的具體步驟(從Java到go),感興趣的朋友跟隨小編一起學習下吧2022-09-09Golang編程并發(fā)工具庫MapReduce使用實踐
這篇文章主要為大家介紹了Golang并發(fā)工具庫MapReduce的使用實踐,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-04-04