laravel中间件的创建思路分析

网上有很多解析laravel中间件的实现原理,但是不知道有没有读者在读的时候不明白,作者是怎么想到要用array_reduce函数的?

推荐:laravel教程

本文从自己的角度出发,模拟了如果我是作者,我是怎么实现这个中间件功能,又是怎么找到并使用对应的函数。

什么是laravel中间件

Laravel 中间件提供了一种机制在不修改逻辑代码的情况下,中断原本程序流程,通过中间件来处理一些事件,或者扩展一些功能。比如日志中间件可以方便的记录请求和响应日志,而不需要去更改逻辑代码。

那么我们简化一下软件执行过程,现在有一个核心类kernel,下面是它的laravel代码

#捕获请求$request = Illuminate\Http\Request::capture()#处理请求$response = $kernel->handle($request);

代码的作用是 捕获一个 Request ,返回一个 Response。这里面就是后续分发到具体执行逻辑的代码段并返回结果。

那么如果想在执行这个$kernel->handle()方法之前或者之后,增加一段逻辑一般会怎么写呢。大概如下:

$request = Illuminate\Http\Request::capture()function midware(){    before()#在之前执行的语句集合    #####        $response = $kernel->handle($request);    #####    after()#在之后执行的语句集合}

显然这样写没有问题,但是毫无拓展性可言,想执行什么东西都要更改这个方法,这种是不可能封装成框架核心内容的。怎么改进呢

定义一个要执行的中间件类叫middleware,类实现两个方法,before()和after()然后代码如下。

#配置项中有一项配置中间件:middleware = '';$request = Illuminate\Http\Request::capture()function midware(){    middleware.before()    #####        $response = $kernel->handle($request);    #####    middleware.after()}

是否解决了问题呢,是解决了不用更改的问题,但是我们如果需要多个中间件怎么办呢,最容易想到的就是:定义一个中间件数组middleware_arr,每一个middleware类都含有before和after方法,代码如下:

#配置项中有middleware_arrmiddleware_arr=array();$request = Illuminate\Http\Request::capture()function midware(){    foreach(middleware_arr as middleware){       middleware.before()    }    #####        $response = $kernel->handle($request);    #####    foreach(middleware_arr as middleware){        middleware.after()    }}

虽然有点老土,但是的确解决了问题。但是这个还存在一个问题,就是我们怎么向中间件传递参数的问题,那么如下可以吗:

$request = Illuminate\Http\Request::capture()function midware(){    foreach(middleware_arr as middleware){       middleware.before($request)    }    #####        $response = $kernel->handle($request);    #####    foreach(middleware_arr as middleware){        middleware.after($response)    }}

看似是解决了问题,但是仔细分析,就会发现,这里面每次给中间件的都是最初的$request,这显然不行,修改成如下:

$request = Illuminate\Http\Request::capture()function midware(){    foreach(middleware_arr as middleware){       $request = middleware.before($request)    }    #####        $response = $kernel->handle($request);    #####    foreach(middleware_arr as middleware){        $response = middleware.after($response)    }}

还有一个问题就是,假设有两个中间件A和B,那么执行顺序应该是怎么样呢:

$request = Illuminate\Http\Request::capture()$request = A.before($request);$request = B.before($request);$response = $kernel->handle($request);$response = A.after();$response = B.after();

这样合理吗?不太好分辨,我们假设有一个记录请求和响应日志的中间件,这个时候,不论你把它放在什么位置,都不能完美的记录最初请求和最终日志。难道类似情况要写两个类,一个记录请求放在中间件数组第一个,一个处理响应,放在数组最后一位吗?不如在执行后面的foreach之前把middleware_arr数组给反转一下,这样就符合了要求:

$request = Illuminate\Http\Request::capture()$request = A.before($request);$request = B.before($request);$response = $kernel->handle($request);$response = B.after();$response = A.after();

但是我也开始怀疑这个老土且不灵活的方案是否有更好的解决办法,在观察这个执行顺序的时候,发现是一个包裹样式(洋葱式)的。那个接下来的问题就能不能找到更灵活精美的解决方案,看上面这种结构,总感觉有点熟悉,他很像是A的函数包裹B的函数,B的函数包括了最初的执行代码。函数内部调用函数容易,但是咱们这里每一个中间件之间是不知道对方存在的,所以要把其他中间件要执行的函数传递到上一级,这里就用到了闭包函数还有一个php函数array_reduce(),

array_reduce函数定义:mixed array_reduce ( array $input , callable $function [, mixed $initial = NULL ] )

    #输出:这将使 $b  的值为 15, $c  的值为 1200(= 10*1*2*3*4*5)

array_reduce() 将回调函数 function 迭代地作用到 input 数组中的每一个单元中,从而将数组简化为单一的值。咱们是把多个函数包裹成最终调用一个函数。

#我们先假设只有一个middleware,叫log来简化情况,这里的类应该是一个类全路径,我这里就简单的写一下,要不然太长了。    $middleware_arr = ['log'];#最终要执行的代码先封装成一个闭包,要不然没有办法传递到内层,如果用函数名传递函数的话,是没有办法传递参数的。    $default = function() use($request){        return $kernel->handle($request);    }    $callback = array_reduce($middleware_arr,function($stack,$pipe) {        return function() use($stack,$pipe){          return $pipe::handle($stack);        };    },$default);        # 这里 callback最终是 这样一个函数:    function() use($default,$log){          return $log::handle($default);        };        #所以每一个中间件都需要有一个方法handle方法,方法中要对传输的函数进行运行,类似如下,这里我类名就不大写了    class log implements Milldeware {        public static function handle(Closure $func)        {            $func();        }    }    #这里不难看出可以加入中间件自身逻辑如下: class log implements Milldeware {        public static function handle(Closure $func)        {            #这里可以运行逻辑块before()            $func();            #这里可以运行逻辑块after()        }    }

这样在执行callback函数的时候,执行顺序如下:

先运行log::haddle()方法,

执行了log::before()方法

运行default方法,执行$kernel->handle($request)

运行log::after()方法

然后模拟多个的情况如下:

    $middleware_arr = ['csrf','log'];#最终要执行的代码先封装成一个闭包,要不然没有办法传递到内层,如果用函数名传递函数的话,是没有办法传递参数的。    $default = function() use($request){        return $kernel->handle($request);    }    $callback = array_reduce($middleware_arr,function($stack,$pipe) {        return function() use($stack,$pipe){          return $pipe::handle($stack);        };    },$default);        # 这里 callback最终是 执行这样:    $log::handle(function() use($default,$csrf){                    return $csrf::handle($default);                });

执行顺序如下:

1.先运行log::haddle(包含csrf::handle闭包函数)方法,

2.执行了log::before()方法

3.运行闭包也就是运行了$csrf::handle($default)

4.执行了csrf::before()方法

5.运行default方法,执行$kernel->handle($request)

6.执行了csrf::after()方法

7.运行log::after()方法

注意这里还有一个问题就是中间件产生的结果,并没有进行传递,可以通过修改共有资源的方式来达到相同的目的,并非需要真的传值到下一个中间件。

到此这篇文件就结束了,其实其中很多关节都是我写这篇文章的时候才想明白的。尤其是对闭包函数的运用和理解更深了,闭包函数可以延迟利用资源,比如当前不适合执行的语句,又要传递到后面,利用闭包可以封装起来传递出去,这是传统函数做不到的。

以上就是laravel中间件的创建思路分析的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/92812.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月19日 01:50:40
下一篇 2025年11月19日 02:31:25

相关推荐

  • 如何设计XML的异常处理

    XML异常处理需在数据生命周期各环节预设应对策略,通过XML Schema或DTD进行早期验证,解析器捕获格式与结构错误,业务层校验规则,并统一错误报告与恢复机制,构建多层次、可扩展的防御体系。 设计XML的异常处理,说到底,就是要在XML数据生命周期的各个环节——从它的生成、传输到最终的解析和业务…

    2025年12月17日
    000
  • XSD复杂类型如何定义?

    XSD复杂类型用于描述包含多个元素、属性或混合内容的结构化数据,通过定义,可包含序列(sequence)、选择(choice)、全部(all)等内容模型,并支持属性、简单内容扩展及属性组复用,与仅表示原子值的简单类型相比,复杂类型能表达更丰富的数据结构和语义关系。 (选择):在定义的多个子元素中,只…

    2025年12月17日
    000
  • RSS如何实现自动化发布?

    要实现RSS自动化发布,首先选择支持RSS的CMS或使用第三方工具生成XML文件,配置feed信息并确保内容更新时自动同步;订阅者通过RSS阅读器获取信息,可选用Feedly、Reeder等跨平台工具,根据界面、功能、平台和价格选择合适阅读器;遇到订阅源无法访问、内容不更新或重复等问题时,检查URL…

    2025年12月17日
    000
  • XML处理如何避免阻塞?

    核心在于采用流式解析与异步处理结合的方式。首先,放弃DOM这种全量加载模式,改用SAX或StAX实现边读边解析,仅保留当前节点信息,大幅降低内存占用并避免初始化阻塞。其次,在解析过程中将耗时业务逻辑(如数据库写入、复杂计算)封装为任务提交至线程池,实现解析与处理的并行化,防止主线程卡顿。SAX为事件…

    2025年12月17日
    000
  • XPath如何选择后代节点? XPath遍历后代节点的路径写法与实例解析

    XPath选择后代节点主要通过//操作符、/操作符和descendant::轴实现。//用于全局搜索所有匹配节点,如//div选择所有div元素;/用于精确路径选择,如/div/p选择div下的直接子节点p;descendant::轴显式选择所有后代,如div/descendant::p。处理复杂嵌…

    2025年12月17日
    000
  • XPath如何匹配多个节点?

    XPath能匹配多个节点,通过标签名、属性、位置、通配符及联合操作符|等方式实现。例如//a选所有链接,//div[@class=’product-item’]选特定class的div,//h1|//h2|//h3选多种标题。使用谓词可精确筛选,如//div[contains…

    2025年12月17日
    000
  • RSS扩展元素有哪些?

    RSS扩展元素通过XML命名空间机制弥补核心规范的不足,支持播客、视频等复杂内容的元数据描述,如iTunes RSS定义播客作者、封面、时长,Media RSS描述媒体文件属性,content:encoded嵌入完整HTML内容,Dublin Core提供通用元数据,GeoRSS添加地理位置,使RS…

    2025年12月17日
    000
  • XPath如何选择父节点?

    在XPath中选择父节点主要用..或parent::轴,..是parent::node()的简写,两者功能等价但..更简洁常用;parent::可明确指定父节点类型如parent::div,适合需清晰语义的场景;结合谓词可精确筛选父节点,如//a[text()=’Link 2&#8242…

    2025年12月17日
    100
  • RSS订阅如何异常监控?

    答案:RSS订阅异常监控需建立正常基线,通过持续比对更新频率、内容结构、条目数量、HTTP状态等维度发现偏差,并结合分级告警与重试机制避免误报。具体可采用自定义脚本(如Python+feedparser)或Serverless架构实现自动化抓取、解析、存储与告警,同时根据历史数据动态调整阈值,以应对…

    2025年12月17日
    000
  • XSLT如何验证输入?

    XSLT在数据验证中扮演“数据质量检查员”角色,通过条件逻辑、类型转换、xsl:assert和xsl:message等机制,在转换过程中实现数据完整性检查,并可生成结构化错误报告或嵌入错误信息,确保数据符合业务规则。 XSLT本身并非一个专门的验证工具,它更擅长转换。但我们完全可以在转换过程中,通过…

    2025年12月17日
    000
  • RSS如何防止垃圾订阅?

    防止RSS垃圾订阅需从源头控制内容、加强访问安全并过滤审核。首先确保CMS干净,利用反垃圾插件如Akismet拦截垃圾评论;其次对用户提交内容实施人工审核与技术过滤结合;再者通过HTTPS加密传输,对私有Feed采用API Key认证,服务器端配置限流与防火墙;最后借助CMS内置机制如评论审核、权限…

    2025年12月17日
    000
  • RSS生成器需要哪些功能?

    一个优秀的RSS生成器需具备灵活的内容源接入、标准的格式输出、高效的更新机制与良好的可配置性。它通过支持数据库、API、网页抓取等方式解析非结构化信息,将内容转换为符合RSS/Atom规范的XML格式,确保GUID唯一、日期准确、避免重复推送。为保障实时性,应优先采用Webhook事件驱动,辅以定时…

    2025年12月17日
    000
  • XML Schema与DTD有什么区别?

    XML Schema在数据类型和命名空间方面显著优于DTD,它提供丰富的内置类型(如整数、日期、布尔值)和自定义类型能力,支持正则表达式约束,确保数据准确性;同时原生支持命名空间,解决元素名称冲突,实现多词汇表融合,提升XML文档的语义精确性、互操作性和模块化设计能力。 XML Schema和DTD…

    2025年12月17日
    000
  • XPath如何选择后代节点?

    XPath中//和descendant::轴的核心区别在于://是descendant-or-self::node()/的简写,包含当前节点自身及所有后代,而descendant::仅选择后代节点不包括自身。//语法简洁常用于全局搜索,如//div查找所有div元素;descendant::语义明确…

    2025年12月17日
    000
  • RSS中enclosure标签怎么用?

    enclosure标签是播客内容分发的核心,它通过在RSS的item中嵌入url、length和type三个属性,使客户端能发现、下载并正确播放音频等媒体文件。其重要性在于实现媒体订阅功能、提供可靠的分发信息(如文件大小和格式)、支持客户端自动化处理,从而构建创作者与听众间的稳定传输管道。为确保可访…

    2025年12月17日
    000
  • 如何实现RSS内容过滤?

    RSS内容过滤的核心是通过关键词、正则表达式或规则筛选有价值信息,提升信噪比。可借助Inoreader等支持黑白名单的高级阅读器实现可视化过滤,适合普通用户;技术用户则可通过Python脚本自建系统,利用feedparser解析并用正则匹配标题、摘要,实现高度个性化控制。自建方案优势在于灵活性与数据…

    2025年12月17日
    000
  • XML如何优化查询性能?

    答案:优化XML查询性能需结合索引、数据转换与原生数据库。首先,通过XPath/XQuery索引减少扫描量;其次,将XML转换为关系表或JSON以利用高效查询引擎;最后,采用原生XML数据库实现存储与查询的深度优化。 XML查询性能优化,核心在于避免对原始、未索引的XML文档进行全量解析和遍历。通常…

    2025年12月17日
    000
  • 如何提取RSS中的媒体内容?

    提取RSS媒体内容需解析XML结构,定位enclosure或media命名空间节点,获取URL、MIME类型等信息,使用流式下载处理音频、视频、图片等不同媒体类型,避免内存溢出,并通过记录GUID或时间戳实现增量更新,同时用异常处理应对网络、解析等错误。 提取RSS中的媒体内容,关键在于解析RSS的…

    2025年12月17日
    000
  • XSLT模板如何编写?

    XSLT模板的核心是通过匹配(match)和应用(apply-templates)机制,利用xsl:template、xsl:value-of、xsl:for-each、xsl:if等元素,结合XPath定位节点,实现XML到HTML或其他格式的声明式转换。 编写XSLT模板,本质上是定义一套规则,…

    2025年12月17日 好文分享
    000
  • XML如何与SVG整合?

    SVG作为基于XML的矢量图形格式,可通过内联嵌入、外部引用、XSLT转换、JavaScript操作和服务器端生成等方式与XML整合,实现数据与图形的动态交互。其在数据可视化中广受欢迎,得益于无限缩放、小文件体积、可访问性强、高交互性及与数据驱动的天然契合。在HTML中嵌入SVG时,内联SVG适合高…

    2025年12月17日
    000

发表回复

登录后才能评论
关注微信