什么是中间件?
在服务端开发中,“中间件”是大家一定会听到的名词。
初学者听起来觉得很高大上,觉得很难,其实“中间件”并不难,对我们学习Go语言来说,使用中间件也没有太多心智挑战。
“中间件”它并没有很严格的定义,但是普遍接受IDC的定义:中间件是一种独立的系统软件服务程序,分布式应用软件借助这种软件在不同的技术之间共享资源,中间件位于客户机服务器的操作系统之上,管理计算资源和网络通信。
比如:
我们今天要重点介绍的“GoFrame网络请求控制中间件”是中间件的一种,它是基于GoFrame框架内部的一种中间件,也是我们这期内容的重点。
在我们WEB开发过程中,像常用的MySQL、Redis、RabbitMQ在系统架构角度也被称为中间件。
从基础架构、系统架构、程序架构等不同的观点来看,中间件的概念是不一样的。我们也不用过于纠结于这个名词的具体解释。
毕竟结合自己的需求,掌握如何使用中间件,远比只搞清楚中间件的解释要有意义的多。
更多介绍大家可以查看:中间件-计算机用语--百度百科[1],不作为我这期内容的重点。
GoFrame中间件/拦截器-基本介绍
GoFrame提供了优雅的中间件请求控制方式,该方式也是主流的WebServer提供的请求流程控制方式,基于中间件设计可以为WebServer提供更灵活强大的插件机制。
经典的中间件洋葱模型:
中间件定义
中间件的定义和普通HTTP执行方法HandlerFunc一样,但是可以在Request参数中使用Middleware属性对象来控制请求流程。
我们拿一个跨域请求的中间件定义来示例说明一下:
可以看到在该中间件中执行完成跨域请求处理的逻辑后,使用r.Middleware.Next()方法进一步执行下一个流程;如果这个时候直接退出不调用r.Middleware.Next()方法的话,将会退出后续的执行流程(例如可以用于请求的鉴权处理)。
中间件类型
中间件的类型分为两种:前置中间件和后置中间件。前置即在路由服务函数调用之前调用,后置即在其后调用。
前置中间件
其定义类似于:
后置中间件
其定义类似于:
按照中间件注册方式划分,又可分为:全局中间件、分组路由中间件。
全局中间件
全局中间件是可以独立使用的请求拦截方法,通过路由规则的方式进行注册,绑定到Server上,由于中间件需要执行请求拦截操作,因此往往是使用"模糊匹配"或者"命名匹配"规则。
全局中间件仅对动态请求拦截有效,无法拦截静态文件请求。
分组路由中间件
分组路由中注册的中间件绑定到当前分组路由中的所有的服务请求上,当服务请求被执行前会调用到其绑定的中间件方法。
分组路由仅有一个Middleware的中间件注册方法。分组路由中间件与全局中间件不同之处在于,分组路由中间件无法独立使用,必须在分组路由注册中使用,并且绑定到当前分组路由中所有的路由上作为路由方法的一部分。
执行优先级
全局中间件
由于全局中间件也是通过路由规则执行,那么也会存在执行优先级:
- 首先,由于全局中间件是基于模糊路由匹配,因此当同一个路由匹配到多个中间件时,会按照路由的深度优先规则执行,具体请查看路由章节;
- 其次,同一个路由规则下,会按照中间件的注册先后顺序执行,中间件的注册方法也支持同时按照先后顺序注册多个中间件;
- 最后,为避免优先级混淆和后续管理,建议将所有中间件放到同一个地方进行先后顺序注册来控制执行优先级;
这里的建议来参考于gRPC的拦截器设计,没有过多的路由控制,仅在一个地方同一个方法统一注册。往往越简单,越容易理解,也便于长期维护。
分组路由中间件
分组路由中间件是绑定到分组路由上的服务方法,不存在路由规则匹配,因此只会按照注册的先后顺序执行。
示例
官方文档为我们提供了比较详细的示例,包括:
- 允许跨域请求
- 请求鉴权处理
- 鉴权例外处理
- 统一的错误处理
- 自定义日志处理
本文转载自微信公众号「 程序员升级打怪之旅」,作者「王中阳Go」,可以通过以下二维码关注。
转载本文请联系「 程序员升级打怪之旅」公众号。