最近在学习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.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

以上为个人经验,希望能给大家一个参考,也希望大家多多支持。如有错误或未考虑完全的地方,望不吝赐教。

标签:
golang,http重定向

免责声明:本站文章均来自网站采集或用户投稿,网站不提供任何软件下载或自行开发的软件! 如有用户或公司发现本站内容信息存在侵权行为,请邮件告知! 858582#qq.com
狼山资源网 Copyright www.pvsay.com

评论“解决golang http重定向失效的问题”

暂无“解决golang http重定向失效的问题”评论...

《魔兽世界》大逃杀!60人新游玩模式《强袭风暴》3月21日上线

暴雪近日发布了《魔兽世界》10.2.6 更新内容,新游玩模式《强袭风暴》即将于3月21 日在亚服上线,届时玩家将前往阿拉希高地展开一场 60 人大逃杀对战。

艾泽拉斯的冒险者已经征服了艾泽拉斯的大地及遥远的彼岸。他们在对抗世界上最致命的敌人时展现出过人的手腕,并且成功阻止终结宇宙等级的威胁。当他们在为即将于《魔兽世界》资料片《地心之战》中来袭的萨拉塔斯势力做战斗准备时,他们还需要在熟悉的阿拉希高地面对一个全新的敌人──那就是彼此。在《巨龙崛起》10.2.6 更新的《强袭风暴》中,玩家将会进入一个全新的海盗主题大逃杀式限时活动,其中包含极高的风险和史诗级的奖励。

《强袭风暴》不是普通的战场,作为一个独立于主游戏之外的活动,玩家可以用大逃杀的风格来体验《魔兽世界》,不分职业、不分装备(除了你在赛局中捡到的),光是技巧和战略的强弱之分就能决定出谁才是能坚持到最后的赢家。本次活动将会开放单人和双人模式,玩家在加入海盗主题的预赛大厅区域前,可以从强袭风暴角色画面新增好友。游玩游戏将可以累计名望轨迹,《巨龙崛起》和《魔兽世界:巫妖王之怒 经典版》的玩家都可以获得奖励。