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


猜你喜欢
- 在用 Vue 的父子组件传值时遇到一个冷门的问题,子组件改变值后父组件的值也随之改变了,特此记录下原因和解决方式。再系统梳理下 JavaSc
- 基本对比使用方式使用场合优缺点是否需要安装需要的dll网址引用方式程序内引用程序初期确定使用MySql,前期添加引用大多数情况下使用在类文件
- 引言什么是数据结构?数据结构是指相互之间存在着一种或多种关系的数据元素的集合和该集合中数据元素之间的关系组成。简单来说,数据结构就是设计数据
- Python正则表达式中的'r'最近遇到一个非常困惑的地方,那就是在使用Python中的正则表达式的时候,正则表达式前面的&
- 下半年毕业季,各种毕业设计纷涌而至,虽然学校教了Python,但是我相信各位肯定没有认真学,做作业就懵逼的那种~所以我都给大家准备好了!各种
- form 表单中的button按钮<button>提交</button> :放在form表单中,会有一个提交事件,会
- 八皇后问题描述:在一个8✖️8的棋盘上,任意摆放8个棋子,要求任意两个棋子不能在同一行,同一列,同一斜线上,问有多少种解法。规则分析:任意两
- 一.vue路由传值在日常使用vue操作时,某些场景需要用到路由之间的传值,就是在使用router跳转到另外一个路由时需要携带参数一并传过去,
- 限制只能输入数字// -----------------------------------------------------------
- 本文记录了mysql 8.0.17 安装配置的图文教程,供大家参考,具体内容如下1.下载安装包下载地址:安装包2.解压安装包到目录这里解压到
- 其中用到urllib2模块和正则表达式模块。下面直接上代码:[/code]#!/usr/bin/env python#-*- coding:
- 昨天修改机顶盒上面的EPG页面,遇到一个小问题。用户购买游戏时需要弹出购买确认对话框,对话框的默认焦点规定必须停留在“取消”按键上。很简单的
- logging模块是Python的一个标准库模块,开发过程中,可以通过该模块,灵活的完成日志的记录。logging模块提供了两种记录日志的方
- 决定数据类型的第一步是定义所存数数据的分类: 数值型, 字符串型还是临时型等;除了一些特别的并不是那么直观的外, 这通常是很直观的。接下来是
- 网站上的Banner条,是网站用来作为盈利或者是发布一些重要的信息的工具。但是它又不能作为网页的主要内容,因为它的主要目的是吸引人的注意力,
- Python 网页解析HTMLParse的实例详解使用python将网页抓取下来之后,下一步我们就应该解析网页,提取我们所需要的内容了,在p
- 本文实例讲述了Python双精度浮点数运算并分行显示操作。分享给大家供大家参考,具体如下:#coding=utf8def doubleTyp
- 大家是否经常遇到在关闭网页的时候,会看到一个确定是否离开当前页面的提示框?想一些在线测试系统、信息录入系统等就经常会有这一些提示,避免用户有
- 元组的结构在这一小节当中主要介绍在 python 当中元组的数据结构:typedef struct { PyObj
- 本文实例讲述了Python基于pygame模块播放MP3的方法。分享给大家供大家参考,具体如下:安装pygame(可参考:安装Python和