最近在學(xué)習(xí)GoLang,在使用http重定向的時候發(fā)現(xiàn)了一個很有趣的現(xiàn)象,在這里記錄一下。
r.GET("/index", func(c *gin.Context) {
c.Redirect(http.StatusMovedPermanently, "http://www.baidu.com/")
})
本來寫了這么一段代碼,將我的路由重定向到“百度”,第一次試驗成功了。之后當(dāng)我想重新定向到其它網(wǎng)站,或者不重定向而試驗其它邏輯的時候,發(fā)現(xiàn)在瀏覽器中,永遠(yuǎn)只是定向到“百度”。非常疑惑,明明程序都重新運(yùn)行了呀。后來我發(fā)現(xiàn),用postman或者另一個瀏覽器打開,我新寫的邏輯是能實現(xiàn)的。
最后清空了瀏覽器近一小時的記錄,發(fā)現(xiàn)能實現(xiàn)新邏輯了。
所以應(yīng)該是因為瀏覽器緩存導(dǎo)致直接在緩存中取內(nèi)容,而不是從我服務(wù)端。
補(bǔ)充:golang不想http自動處理重定向的解決方案
前言
有時候發(fā)送http請求不想讓庫自動幫忙處理重定向,庫里面默認(rèn)的是會把所有重定向都完成一遍,結(jié)果就是最后一個沒有重定向的請求的結(jié)果。
因此需要一種方案直接獲取首次訪問的結(jié)果,不走重定向。
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 /* 不進(jìn)入重定向 */
},
}
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]))
}
}
// 下面開啟一個服務(wù),重定向到百度
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()))
}
結(jié)論
如下的第一個請求是直接返回百度的網(wǎng)頁,及重定向以后的內(nèi)容。第二個請求直接返回第一個302重定向的內(nèi)容。
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
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。如有錯誤或未考慮完全的地方,望不吝賜教。
您可能感興趣的文章:- 解決golang http.FileServer 遇到的坑
- golang HTTP 服務(wù)器 處理 日志/Stream流的操作
- golang http請求封裝代碼
- 解決golang處理http response碰到的問題和需要注意的點
- 解決golang讀取http的body時遇到的坑
- Golang發(fā)送http GET請求的示例代碼
- 在Golang中使用http.FileServer返回靜態(tài)文件的操作