解决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自带的库,后来发现更简单的方案。
iferr==ErrUseLastResponse{ returnresp,nil//这里是拦截重定向,如果不拦截则走下面的重定向判断 } varshouldRedirectbool redirectMethod,shouldRedirect,includeBody=redirectBehavior(req.Method,resp,reqs[0]) if!shouldRedirect{ returnresp,nil }
解决方案
下面代码可以验证自动处理重定向,以及不走重定向的方案。
packagemain import( "io/ioutil" "log" "net/http" "time" ) funcmain(){ goserver() time.Sleep(time.Second) mUrl:="http://127.0.0.1:12345/post" {//常规方法 req,err:=http.NewRequest(http.MethodPost,mUrl,nil) iferr!=nil{ log.Fatal(err) } resp,err:=http.DefaultClient.Do(req) ifresp!=nil{ deferresp.Body.Close() } iferr!=nil{ log.Fatal(err) } byt,err:=ioutil.ReadAll(resp.Body) iferr!=nil{ log.Fatal(err) } log.Println(resp.StatusCode,"|",string(byt[:128])) } {//去掉自动处理重定向 req,err:=http.NewRequest(http.MethodPost,mUrl,nil) iferr!=nil{ log.Fatal(err) } resp,err:=http.DefaultTransport.RoundTrip(req) ifresp!=nil{ deferresp.Body.Close() } iferr!=nil{ log.Fatal(err) } byt,err:=ioutil.ReadAll(resp.Body) iferr!=nil{ log.Fatal(err) } log.Println(resp.StatusCode,"|",string(byt[:128])) } {//另一种不要重定向的方法 req,err:=http.NewRequest(http.MethodPost,mUrl,nil) iferr!=nil{ log.Fatal(err) } client:=&http.Client{ CheckRedirect:func(req*http.Request,via[]*http.Request)error{ returnhttp.ErrUseLastResponse/*不进入重定向*/ }, } resp,err:=client.Do(req) ifresp!=nil{ deferresp.Body.Close() } iferr!=nil{ log.Fatal(err) } byt,err:=ioutil.ReadAll(resp.Body) iferr!=nil{ log.Fatal(err) } log.Println(resp.StatusCode,"|",string(byt[:128])) } } //下面开启一个服务,重定向到百度 funcserver(){ http.HandleFunc("/post",mPost) http.ListenAndServe(":12345",nil) } funcmPost(whttp.ResponseWriter,r*http.Request){ http.Redirect(w,r,"http://www.baidu.com",http.StatusFound) w.Write([]byte(time.Now().String())) }
结论
如下的第一个请求是直接返回百度的网页,及重定向以后的内容。第二个请求直接返回第一个302重定向的内容。
2020/10/1413:11:56200|百度一下,你就知道
2020/10/1413:11:56302|2020-10-1413:11:56.6559382+0800CSTm=+1.429170501
2020/10/1413:11:56302|2020-10-1413:11:56.6559382+0800CSTm=+1.429170501
以上为个人经验,希望能给大家一个参考,也希望大家多多支持毛票票。如有错误或未考虑完全的地方,望不吝赐教。