Gin框架中间件原理

先了解闭包

闭包指的是一个函数和与其相关的引用环境组合而成的实体。简单来说,闭包=函数+引用环境

func adder() func(int) int {
	var x int
	return func(y int) int {
		x += y
		return x
	}
}
func main() {
	var f = adder()
	fmt.Println(f(10)) //10
	fmt.Println(f(20)) //30
	fmt.Println(f(30)) //60

	f1 := adder()
	fmt.Println(f1(40)) //40
	fmt.Println(f1(50)) //90
}

Gin框架中中间件

在Gin框架中,中间件(Middleware)是在处理请求的过程中,在请求被真正的处理函数处理之前或之后执行的一些逻辑。

其原理主要是通过函数式编程和Go语言的闭包特性来实现的。

当定义一个中间件时,实际上是定义了一个函数,这个函数接收gin.Context作为参数。例如:

func Logger() gin.HandlerFunc {
    return func(c *gin.Context) {
       // 记录请求开始时间
       t := time.Now()
       // 调用下一个中间件或者处理函数
       c.Next()
       // 计算请求耗时并打印日志
       latency := time.Since(t)
       log.Printf("%s %s %v", c.Request.Method, c.Request.URL.Path, latency)
    }
}

在这个中间件函数内部,通过c.Next()来调用下一个中间件或者最终的处理函数。

当使用Use方法添加中间件到Gin引擎时,Gin会将这些中间件函数组成一个链式结构。当一个请求到来时,Gin会从链头开始逐个执行中间件函数。在每个中间件函数中,可以在c.Next()调用之前进行前置操作,比如验证用户身份、记录日志等;在c.Next()调用之后进行后置操作,比如添加响应头、记录响应时间等。这样就实现了对请求处理过程的拦截和增强。

具体实现

以下是Gin框架将中间件函数组成链式结构的具体实现原理及相关细节:

1. 核心数据结构

在Gin框架中,有一个关键的结构体(简化示意),类似如下:

type Engine struct {
    // 用于存储中间件函数的切片
    middlewares []gin.HandlerFunc
    // 路由树等其他相关结构,这里重点关注中间件相关部分
    //...
}

Engine结构体中的middlewares切片就是用来存放通过Use方法添加进来的中间件函数的,每个中间件函数都是gin.HandlerFunc类型,它的定义本质上是一个接收*gin.Context作为参数并返回void的函数类型,例如:

type HandlerFunc func(*gin.Context)

2. Use方法的作用

当调用EngineUse方法来添加中间件时,代码逻辑大致如下(简化版):

func (engine *Engine) Use(middleware...gin.HandlerFunc) {
    engine.middlewares = append(engine.middlewares, middleware...)
}

Use方法就是简单地将传入的中间件函数一个个追加到middlewares切片中,这样就完成了中间件的收集,为后续构建链式结构做准备。

3. 构建链式结构(核心在于请求处理流程)

当有请求进来时,Gin会从已添加的中间件链头开始逐个执行中间件函数,这个过程是在处理请求的主逻辑中实现的。在Gin内部处理请求的函数中(简化示意如下):

func (engine *Engine) ServeHTTP(w http.ResponseWriter, r *http.Request) {
    c := engine.pool.Get().(*gin.Context)
    c.Writer = &responseWriter{Writer: w}
    c.Request = r

    // 遍历中间件切片,构建链式调用
    h := engine.combineHandlers()
    h(c)

    engine.pool.Put(c)
}

关键在于engine.combineHandlers()这个方法,它会将middlewares切片中的中间件函数组合起来,返回一个最终的处理函数(也是gin.HandlerFunc类型),组合的方式大概是通过闭包和函数嵌套来实现类似如下逻辑(简化示意,实际更复杂):

func (engine *Engine) combineHandlers() gin.HandlerFunc {
    return func(c *gin.Context) {
        var currentIndex int
        var nextHandler func()
        // 定义下一个中间件或最终处理函数的执行逻辑
        nextHandler = func() {
            if currentIndex < len(engine.middlewares) {
                middleware := engine.middlewares[currentIndex]
                currentIndex++
                // 调用当前中间件,并在当前中间件内部调用nextHandler来执行后续的中间件或最终处理函数
                middleware(c)
            } else {
                // 如果中间件都执行完了,执行路由对应的最终处理函数(比如某个具体的控制器方法)
                engine.handleContext(c)
            }
        }
        // 启动链式调用,从第一个中间件开始
        nextHandler()
    }
}

在上述代码中:

  • 通过定义nextHandler函数,它会根据当前中间件的索引来决定是调用下一个中间件还是执行最终的路由处理函数(engine.handleContext(c)部分,这里省略其具体细节,它主要涉及根据路由匹配去执行对应的业务逻辑处理函数)。
  • 每个中间件函数内部通过调用c.Next()其实就是间接调用了这个nextHandler函数,从而实现了控制权在中间件之间以及最终处理函数之间的有序转移,构建起了链式结构,使得请求在处理过程中会依次经过各个中间件,并且在合适的时机执行后续操作或者返回响应。

总的来说,Gin借助切片收集中间件函数,然后在请求处理阶段通过巧妙的函数嵌套、闭包以及对函数执行顺序的控制,实现了将中间件函数组成链式结构来处理请求的机制。

理解gin中间件和go直接作为http server的中间件有什么区别?

Gin中间件实现原理总结

  • Gin中间件是基于Go语言的函数式编程和闭包来实现的。它主要围绕gin.Context进行操作。gin.Context包含了请求和响应的所有信息,如请求方法、请求路径、请求头、响应状态码等。
  • 链式调用:Gin通过将多个中间件函数组合成一个链来工作。当一个请求到达时,它会按添加中间件的顺序依次执行这些中间件。每个中间件函数可以选择在调用c.Next()之前执行一些前置操作,如权限验证、日志记录开始时间等。
  • 控制权转移c.Next()函数是关键。当执行到c.Next()时,它会暂停当前中间件的执行,将控制权转移给下一个中间件或者最终的处理函数。在所有中间件(以及最终处理函数)执行完返回后,c.Next()之后的代码会继续执行,用于完成后置操作,如记录响应时间、修改响应头等。

与Go直接作为http server中间件的区别

  • 抽象程度不同
    • Go原生http中间件:在Go原生的http包中写中间件,需要直接操作http.Requesthttp.ResponseWriter。这两个对象提供了相对底层的对HTTP请求和响应的操作接口,开发者需要自己处理诸如读取请求头、写入响应状态码等细节。
    • Gin中间件:Gin的中间件基于gin.Context,它对http.Requesthttp.ResponseWriter进行了封装,提供了更高级、更方便的API。例如,可以通过c.Query("param_name")直接获取URL查询参数,而不用像在原生http中那样手动解析。
  • 中间件组织形式不同
    • Go原生http中间件:在Go原生http中,若要构建中间件链,通常需要自己手动调用下一个中间件函数。这可能会导致代码结构不够清晰,特别是当中间件数量较多时,容易出现混乱。
    • Gin中间件:Gin通过Use方法将中间件添加到引擎,自动构建中间件链。这种方式更加清晰、简洁,使得中间件的管理和复用更加方便。
  • 功能丰富度不同
    • Go原生http中间件:Go原生的中间件功能相对基础,主要聚焦于对HTTP基本操作的拦截和处理。
    • Gin中间件:Gin中间件除了基本的HTTP操作外,还提供了许多与Web开发密切相关的功能。比如,Gin的中间件可以很方便地集成日志记录、错误处理、权限验证等多种功能,并且这些功能可以通过Gin提供的插件或者自定义中间件轻松扩展。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/951267.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Node.js——path(路径操作)模块

个人简介 &#x1f440;个人主页&#xff1a; 前端杂货铺 &#x1f64b;‍♂️学习方向&#xff1a; 主攻前端方向&#xff0c;正逐渐往全干发展 &#x1f4c3;个人状态&#xff1a; 研发工程师&#xff0c;现效力于中国工业软件事业 &#x1f680;人生格言&#xff1a; 积跬步…

【Verdi实用技巧-Part2】

Verdi实用技巧-Part2 2 Verdi实用技巧-Part22.1 Dump波形常用的task2.1.1 Frequently Used Dump Tasks2.1.2 Demo 2.2 提取波形信息小工具--FSDB Utilities2.3 Debug in Source code view2.3.1 Find Scopes By Find Scope form 2.3.2 Go to line in Souce code View2.3.3 Use B…

web-前端小实验4

实现以上图片中的内容 代码 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>用户注册</title&…

NLP项目实战——基于Bert模型的多情感评论分类(附数据集和源码)

在当今数字化的时代&#xff0c;分析用户评论中的情感倾向对于了解产品、服务的口碑等方面有着重要意义。而基于强大的预训练语言模型如 Bert 来进行评论情感分析&#xff0c;能够取得较好的效果。 在本次项目中&#xff0c;我们将展示如何利用 Python 语言结合transformers库&…

TAS测评倍智题库 | 益丰大药房2025年中高层测评BA商业推理测评真题考什么?

您好&#xff01;您已被邀请参加360评估。您的评估与反馈将有助于被评估人更深入地了解个人情况&#xff0c;发现个人优势和潜在风险。请您秉持公正、开放的心态进行评估。请尽快完成评估&#xff0c;在此衷心感谢您的配合与支持&#xff01; ​ 相关事宜&#xff1a; 请您在…

优秀的大模型会不会做坏事?

主要围绕大型语言模型&#xff08;LLMs&#xff09;在特定情境下可能出现的欺骗行为及相关研究展开&#xff0c;具体如下&#xff1a; 研究背景与核心发现&#xff1a;研究发现即使在用户无意激励的情况下&#xff0c;LLMs 也可能说谎&#xff0c;而能使用工具的模型更易被诱导…

fiscoBcos落盘加密介绍

落盘加密 落盘加密是在机构内部进行的&#xff0c;每个机构独立管理自己硬盘数据的安全。内网中&#xff0c;每个节点的硬盘数据是被加密的。所有加密数据的访问权限&#xff0c;通过Key Manager来管理。Key Manager是部署在机构内网内&#xff0c;专门管理节点硬盘数据访问秘…

完全二叉树的删除

&#xff08;1&#xff09;删除叶子节点 找到要删除的节点 targetNode找到要删除节点的父节点parent&#xff08;父节点是否存在&#xff09;要删除的节点是父节点的左子树还是右子树如果是左子树&#xff0c;则parent.leftnull;如果是右子树则parent.rightnull。 &#xff08;…

ModuleNotFoundError: No module named ‘setuptools_rust‘ 解决方案

大家好,我是爱编程的喵喵。双985硕士毕业,现担任全栈工程师一职,热衷于将数据思维应用到工作与生活中。从事机器学习以及相关的前后端开发工作。曾在阿里云、科大讯飞、CCF等比赛获得多次Top名次。现为CSDN博客专家、人工智能领域优质创作者。喜欢通过博客创作的方式对所学的…

【算法】时间复杂度以及O(N^2)的排序

目录 1.常数时间的操作 2.时间复杂度 2.1.以选择排序为例 2.2.O(n^2)从何而来 2.3.冒泡排序 2.3.1.抑或运算 2.4.插入排序 3.二分法 3.1.局部最小 4.递归 4.1.递归行为时间复杂度的估计 1.常数时间的操作 一个操作如果和样本的数据量无关&#xff0c;每次都是固定时…

uni app 写的 小游戏,文字拼图?文字拼写?不知道叫啥

从下方的偏旁部首中选在1--3个组成上面文章中的文字&#xff0c;完成的文字标红 不喜勿喷 《满江红》 其中用到了两个文件 strdata.json parameters.json 这两个文件太大 放到资源中了 资源文件 <template><view class"wenzi_page_main"><view c…

【杂记】qt

1、终端下载PySide6以转换文件格式&#xff1a;pip install PySide6 -i https://pypi.tuna.tsinghua.edu.cn/simple 命令提示符下载完毕后&#xff1a;powerShell &#xff1a;cd 跳转到文件对应地址 &#xff08;1、pyside6-uic.exe test.ui -o test.py #将Ui界面文件转换成…

宁德时代2025年Verify入职测评语言理解及数字推理真题SHL题库汇总、考情分析

宁德时代社招Verify入职测评对薪酬有着重要影响&#xff0c;其规定正确率达到80%才能顺利通过测评。这体现了公司对人才专业素养与能力的严格要求&#xff0c;旨在筛选出真正符合岗位需求的优秀人才。测评内容涵盖了专业知识、技能运用、逻辑思维等多方面&#xff0c;只有综合能…

Jenkins-持续集成、交付、构建、部署、测试

Jenkins-持续集成、交付、构建、部署、测试 一: Jenkins 介绍1> Jenkins 概念2> Jenkins 目的3> Jenkins 特性4> Jenkins 作用 二&#xff1a;Jenkins 版本三&#xff1a;DevOps流程简述1> 持续集成&#xff08;Continuous Integration&#xff0c;CI&#xff0…

Flink系统知识讲解之:如何识别反压的源头

Flink系统知识之&#xff1a;如何识别反压的源头 什么是反压 Ufuk Celebi 在一篇古老但仍然准确的文章中对此做了很好的解释。如果您不熟悉这个概念&#xff0c;强烈推荐您阅读这篇文章。如果想更深入、更低层次地了解该主题以及 Flink 网络协议栈的工作原理&#xff0c;这里有…

Go学习:多重赋值与匿名变量

1. 变量的多重赋值 1.1 基本语法格式 go语言中&#xff0c;可以将多个赋值语句 合并成 一句&#xff0c;比如&#xff1a; a : 10 b : 20 c : 30//a,b,c三个变量的赋值语句可以简练成以下格式a, b, c : 10, 20, 30 1.2 交换变量值 当需要交换两个变量的值时&#…

ArkUI-应用数据持久化

应用数据持久化&#xff0c;是指应用将内存中的数据通过文件或数据库的形式保存到设备上。内存中的数据形态通常是任意的数据结构或数据对象&#xff0c;存储介质上的数据形态可能是文本、数据库、二进制文件等。 HarmonyOS标准系统支持典型的存储数据形态&#xff0c;包括用户…

SOLID原则学习,开闭原则

文章目录 1. 定义2. 开闭原则的详细解释3. 实现开闭原则的方法4. 总结 1. 定义 开闭原则&#xff08;Open-Closed Principle&#xff0c;OCP&#xff09;是面向对象设计中的五大原则&#xff08;SOLID&#xff09;之一&#xff0c;由Bertrand Meyer提出。开闭原则的核心思想是…

西电-算法分析-研究生课程复习笔记

24年秋的应该是张老师最后一次用卷面考试&#xff0c;他说以后这节课的期末考试都是在OJ上刷题了张老师上课还挺有意思的&#xff0c;上完之后能学会独立地思考算法设计问题了。整节课都在强调规模压缩这个概念&#xff0c;考试也是考个人对这些的理解&#xff0c;还挺好玩的哈…

插入实体自增主键太长,mybatis-plaus自增主键

1、问题 spring-boot整合mybtais执行insert语句时&#xff0c;主键id为长文本数据。 2、分析问题 1)数据库主键是否自增 2&#xff09;数据库主键的种子值设置的多少 3、解决问题 1&#xff09;数据库主键设置的时自增 3&#xff09;种子值是1 所以排查是数据库的问题 4、继…