解决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.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

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

(0)

相关推荐

  • Golang HTTP 服务平滑重启及升级的思路

    Golang HTTP服务在上线时,需要重新编译可执行文件,关闭正在运行的进程,然后再启动新的运行进程.对于访问频率比较高的面向终端用户的产品,关闭.重启的过程中会出现无法访问(nginx表现为502)的情况,影响终端用户的使用体验. 实现的一般思路 一般情况下,要实现平滑重启或升级,需要执行以下几个步骤: 发布新的bin文件覆盖老的bin文件 发送一个信号量(USR2),告诉正在运行的进程,进行重启 正在运行的进程接受到信号后,以子进程的方式启动新的bin文件 新进程接收并处理新的请求 老进程

  • Golang Http 验证码示例实现

    验证码(CAPTCHA)是"Completely Automated Public Turing test to tell Computers and Humans Apart"(全自动区分 计算机 和人类的 图灵测试 )的缩写,是一种区分用户是计算机还是人的公共全自动 程序 .可以防止:恶意破解密码. 刷票 .论坛灌水,有效防止某个黑客对某一个特定注册用户用特定程序暴力破解方式进行不断的登陆尝试,实际上用验证码是现在很多网站通行的方式,我们利用比较简易的方式实现了这个功能.这个问题可

  • Golang发送http GET请求的示例代码

    使用标准库http来实现 package tools import ( "io/ioutil" "net/http" ) func Get(url string)string{ res, err :=http.Get(url) if err != nil { return "" } robots, err := ioutil.ReadAll(res.Body) res.Body.Close() if err != nil { return &qu

  • 详解Golang开启http服务的三种方式

    前言 都说go标准库实用,Api设计简洁.这次就用go 标准库中的net/http包实现一个简洁的http web服务器,包括三种版本. v1最简单版 直接使用http.HandleFunc(partern,function(http.ResponseWriter,*http.Request){}) HandleFunc接受两个参数,第一个为路由地址,第二个为处理方法. //v1 func main() { http.HandleFunc("/", func(w http.Respon

  • 详解golang开发中http请求redirect的问题

    这两天在开发项目的时候遇到了一个问题,请求了一个URL,它会302到另一个地址,本意上只是想检查这个URL是否会做3XX的redirect跳转,结果每次reqeust都会返回最后一跳的结果.后来就看了下源码,了解下请求跳转的机制 实现代码 看下实现的简单代码 func main() { client := &http.Client{} url := "http://www.qq.com" reqest, err := http.NewRequest("GET"

  • 详解Golang实现http重定向https的方式

    以前写代码时,都是直接将程序绑定到唯一端口提供http/https服务,在外层通过反向代理(nginx/caddy)来实现http和https的切换.随着上线后的服务越来越多,有一些服务无法直接通过反向代理来提供这种重定向,只能依靠代码自己实现.所以简要记录一下如何在代码中实现http到https的重定向. 分析 无论是反向代理还是代码自己实现,问题的本质都是判断请求是否是https请求. 如果是则直接处理,如果不是,则修改请求中的url地址,同时返回客户端一个重定向状态码(301/302/30

  • 解决golang http重定向失效的问题

    最近在学习GoLang,在使用http重定向的时候发现了一个很有趣的现象,在这里记录一下. r.GET("/index", func(c *gin.Context) { c.Redirect(http.StatusMovedPermanently, "http://www.baidu.com/") }) 本来写了这么一段代码,将我的路由重定向到"百度",第一次试验成功了.之后当我想重新定向到其它网站,或者不重定向而试验其它逻辑的时候,发现在浏览器

  • 解决vue热替换失效的根本原因

    新手刚开始使用vue时,常会遇见一个坑,那就是热替换失效. 什么?你跟我说使用官方的vue-cli去构建,我就是使用vue-cli后突然失效. 什么?你跟我说重新npm run dev一下,好嘛,已经run了N次了依然没回到大路上. 经过在网上一番查找,发现基本没有这个问题的详解,可能是这个问题太低级了? 讲解一下热替换的原理:热替换是在执行npm run dev后,会启动一个本地服务器(webpack-dev-server),这个服务器会观察源代码编译出来的文件.一旦修改了源代码,就会立刻编译

  • 解决layui弹框失效的问题

    之前一直用layer弹框组件一直出现这个 layer.js:2 Uncaught TypeError: i is not a function at Object.o.run (layer.js:2) at layer.js:2 at layer.js:2 at layer.js:2问题 后来发现header头部引入js和css文件调换一下就好了,之间文件样式是这样的 <script src="__PUBLIC__/layer/layer.js"></script&g

  • 解决layui中onchange失效以及form动态渲染失效的问题

    最近在使用layui框架做城市的三级级联的时候,发现jquery中的onchange事件方法在layui框架中失效了. 然后浏览layui社区,了解到layui监控一个控件变化有自己的方法. html <select lay-filter="test"></select> js var form = layui.form(); form.on('select(test)', function(data){ console.log(data); }); 这样操作即

  • 解决VUE双向绑定失效的问题

    双向绑定失效的原因有很多. lz就说最近遇到的. 是的,单价下的那个输入框我用了双向绑定(比如叫price,比如100).然后ipnut 键入中文时,(即使我做了输入验证).回车时虽然框中不会保留中文,但事实上VUE的双向绑定已经失效了.不管你后面输入什么,绑定的price保存的值只会是中文前的那个值(100). 这样就导致 表面好像没事,但是当你提交时就数据不对了. 还有一种是. 是日期控件(bootstrap的)的问题,部分控件才会.mmp哦,当lz用这个空间选定日期时.以为美滋滋的一次搞定

  • 快速解决element的autofocus失效问题

    原因: autofocus是vue中input的原生属性,element也支持这种方法, 但是element中的el-input组件外面还有其他组件, 导致autofocus失效, 只能手动调用focus方法来聚集. 方法: 绑定ref <el-input ref="myNameId" v-model="form.name" /> this.$refs.myNameId.focus(); 动态绑定ref <el-input :ref="

  • 解决 Golang VS Code 插件下载安装失败的问题

    最近开始学Go语言,但是在使用VS Code 编写Go的时候出现了插件无法下载的问题.最初我的解决办法也是从github下载再安装,但是我并不喜欢这种做法,因为我要在多台pc上使用VS Code编写Go,所以我觉要重复多次很麻烦,而且插件的安装也非常麻烦,我曾经一度想放弃学习Go语言,没错因为安装插件很麻烦.在经过我多次的查找最终找到了http://goproxy.cn/,这是Go的国内镜像,使用了这个就可以结局Go插件无法下载的问题. (http://goproxy.cn/ 文档简介,方便以后

  • IDEA解决springboot热部署失效问题(推荐)

    一.什么是热部署? 热部署,就是在应用正在运行的时候升级软件,却不需要重新启动应用. 二.什么是SpringBoot热部署? SpringBoot热部署就是在项目正在运行的时候修改代码, 却不需要重新启动项目. 有了SpringBoot热部署后大大提高了开发效率,因为频繁的重启项目,势必会浪费很多时间, 有了热部署后,妈妈再也不用担心我修改代码重启项目了~~~ 下面看下IDEA解决springboot热部署失效问题,IDEA实现springboot热部署详情如下: 在pom.xml文件中添加依赖

  • 完美解决golang go get私有仓库的问题

    解决golang go get gitlab私有仓库的问题(1.13) 1. 问题描述 require ( git.xxxxxxx.com/middle/user v0.0.1 ) go mod tidy 导入包失败 go get git.xxxxxxx.com/middle/user 失败 go build 有CHECKSUM过程,无法编译 2. 现象分析 go get 不支持代码支持之外的仓库.并且git 调用链过程采取了https 下载过程如果机器设置了GOPROXY,会导致下载失败 编译

  • 解决Golang在Web开发时前端莫名出现的空白换行

    问题: 在使用Golang做Web开发时,有时候渲染出来的模板在前台显示时会出现一些奇怪的空白换行,具体特征就是查看css样式表并没有相关定义的空白部分. 分析: 查看出现问题页面的网页源代码,复制空白换行部分转码为Unicode编码,发现其中包含了\u2028这样的字符[图一].  [图一] 另外,直接在浏览器检查里边的html选择Edit as html,也可发现在空白处存在小红点,鼠标移上会直接显示该字符是\u2028[图二].查阅资料发现,这样的换行符在JavaScript中并不支持.

随机推荐