Golang Gin 中间件 Next()方法示例详解
作者:富士康质检员张全蛋 发布时间:2024-02-20 07:29:22
Next()方法
在中间件中调用 next() 方法,会从 next() 方法调用的地方跳转到
Handler函数Handler函数执行完成,若中间件还有部分代码未执行(中间件中next()之后的代码),则执行该代码
如果第一个中间件处理直接使用了next函数,直接跳转到handler函数,处理完之后还会回过来处理剩下的中间件函数。
除了最中心的这块,其他的都是中间件,当我一个中间件进来,当我直接使用了next之后,直接跳到业务处理这部分的handler,然后业务处理的handler处理完毕之后,然后它再回跳到一层一层未处理完的这个中间件代码。
所以它就非常像洋葱,它其实中间件是一层一层,但是前面只执行了一半,处理完中间件之后是响应了,响应的时候还会再去处理剩下的这个中间件一层一层。
// 日志中间件
func MiddleWareLog() gin.HandlerFunc {
//这里也是context,因为你的中间件也是对请求和响应的处理,只要涉及到请求和响应的处理就用到context上下文
return func(c *gin.Context) {
fmt.Println("log start")
c.Next()
fmt.Println("log end")
}
}
func MiddleWareRequestId() gin.HandlerFunc {
return func(c *gin.Context) {
fmt.Println("request id start")
c.Next()
fmt.Println("request id end")
}
}
func main() {
r := gin.Default()
r.Use(MiddleWareLog(), MiddleWareRequestId())
r.GET("/", func(c *gin.Context) {
fmt.Println("app handler.......")
c.JSON(http.StatusOK, gin.H{
"msg": "success",
})
})
r.Run(":8000")
}
[GIN-debug] Listening and serving HTTP on :8000
log start
request id start
app handler.......
request id end
log end
[GIN] 2023/04/08 - 09:51:03 | 200 | 2.2505ms | ::1 | GET "/"
首先可以看到我先注册了log中间件,然后注册了requestid中间件的id。所以说是按照注册中间件的顺序来的,然后会执行完。
(1)如果上面中间件没有加入next函数,那么它会将中间件里面所有逻辑执行完
(2)如果只有一层中间件,那么在中间件中调用 next() 方法,会从 next() 方法调用的地方跳转到Handler函数。
如果有多层中间件,那么会跳转到链表里面下一个中间件,上面就是从log中间件跳转到request中间件了。在requestid中间件执行了next函数,它下面没有中间件了,只有业务代码了,那么就去执行业务的handler,并且执行好了之后我返回出去,最后执行剩下的中间件那部分没执行完的逻辑。
其实和洋葱一样,最开始执行的是最外面的一层,后执行的是最里面的一层。但是业务代码执行完了之后,我从最里面这层开始
业务场景:最里面这一层的中间件,就是requestid了,可以去做一个开始的计时,next()然后里面去执行业务代码,执行完之后返回里面再来一个结束的计时,那么就能够拿到这个方法的执行时间。
在写一些微服务框架,做一个metric的展现的时候,其实都是借助中间件这么做的,然后将数据暴露给一个接口,然后Prometheus去拿一下,最后展现出来,就是这样实现的。
实现token认证
http://127.0.0.1:8080/index index首页无需token直接访问
http://127.0.0.1:8080/home home家目录需要对token进行验证,验证通过才可访问
中间件最大的存储长度是60多,不能再往下存了。
c.abort()就是直接跳到链表最后一个, 后面没有了,那么也就执行完了,其他的中间件就不执行了。
package main
import (
"fmt"
"github.com/gin-gonic/gin"
"net/http"
)
func AuthMiddleWare() gin.HandlerFunc {
return func(c *gin.Context) {
//常用场景,客户端也就是接口的调用方,客户端携带token的方式 1:请求头(最常用) 2:请求体 3:uri
token := c.Request.Header.Get("token")
fmt.Println("获取token信息:", token)
if token != "qazw1234" {
c.JSON(http.StatusOK, gin.H{
"msg": "身份验证不通过",
})
//Abort方法用于跳到handler链表最后一个index,相当于直接退出,不执行后面的handler了
c.Abort()
return
}
}
}
func main() {
r := gin.Default()
r.GET("/home", AuthMiddleWare(), func(c *gin.Context) {
c.JSON(http.StatusOK, gin.H{
"msg": "home路由",
})
})
r.Run(":8000")
}
来源:https://blog.csdn.net/qq_34556414/article/details/130024569
猜你喜欢
- 有如下 Pandas DataFrame:import pandas as pdinp = [{'c1':10, '
- 本文实例讲述了Python利用神经网络解决非线性回归问题。分享给大家供大家参考,具体如下:问题描述现在我们通常使用神经网络进行分类,但是有时
- 最近在抓取http://skell.sketchengine.eu网页时,发现用requests无法获得网页的全部内容,所以我就用selen
- 前言:用xtarbackup来同步数据,然后基于GTID来设置主从。一、用xtarbackup备份数据库1.1 优势使用xtarbackup
- 一、字符编码简史:美国:1963年 ASCII (包含127个字符 占1个字节)中国:1980年 GB2312 (收录7445个
- 实验室导师又有任务,从一个.dat文件中读取波形数据,通过一个自编码网络进行异常检测。可是对我这种小白来说真的艰难。从最基础的文件读写开始吧
- APScheduler (advanceded python scheduler)是一款Python开发的定时任务工具。文档地址
- 一般来说,内置的slice()函数会创建一个切片对象,可以用在任何允许进行切片操作的地方。下面是slice的简介:# slice 两种用法c
- 我们知道,在调用函数时如果不指定某个参数,Python 解释器会抛出异常。为了解决这个问题,Python 允许为参数设置默认值,即在定义函数
- 例子class A(object): def foo(self,x): print "exe
- 有过一定的 Python 经验的开发者都知道,当引入第三方包时,我们常常会使用 pip install 命令来下载并导入包。那么,如何写一个
- 前言: 在项目开发中,一些业务表字段经常使用日期和时间类型,而且后续还会牵涉到这类字段的查询。关于日期及时间的查询等各类需求也很多,本篇文章
- 看到很多站长工具网,都提供了通过域名获取网站IP的方法。自己也想做一个,网上查了不少代码。有说用WSHSHELL,也有说用ASPPING组件
- 由于最近在处理shp文件,想要跳出arcpy的限制,所以打算学习一下pyshp包的使用方法。在使用《Python地理空间分析指南(第2版)》
- 一、 概念: ① 数据库同步 (主从同步 --- 主数据库写的同时 往从服务器写数据)② 数据库同步 (主主同步 -
- 本文实例讲述了wxPython窗口的继承机制,分享给大家供大家参考。具体分析如下:示例代码如下:import wx class
- 以下各种方式仅供参考,本人亲测只有官方提供的方式比较靠谱。1. 使用多个进程启动多个Tornado实例import tornado.http
- 核心思想在defer出现的地方插入了指令CALL runtime.deferproc,在函数返回的地方插入了CALL runtime.def
- 前言: 最近在学习过程中总是遇到np.random.seed()这个问题,刚开始总是觉得不过是一个简单的随机数种子,就没太在意,后来遇到的次
- Git 恢复到之前版本1. 应用场景进行了错误提交,需要将代码回退至某个版本;或者需要检出某个版本的代码,再切换回最新版本。2. 解决方法2