欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

淺談laravel中間件的創(chuàng)建思路

 更新時(shí)間:2021年05月29日 14:23:37   作者:八重櫻  
網(wǎng)上有很多解析laravel中間件的實(shí)現(xiàn)原理,但是不知道有沒有讀者在讀的時(shí)候不明白,作者是怎么想到要用array_reduce函數(shù)的?本文從自己的角度出發(fā),模擬了如果我是作者,我是怎么實(shí)現(xiàn)這個(gè)中間件功能,又是怎么找到并使用對(duì)應(yīng)的函數(shù)。本文將介紹Laravel中間件得創(chuàng)建思路。

Laravel 中間件提供了一種機(jī)制在不修改邏輯代碼的情況下,中斷原本程序流程,通過中間件來處理一些事件,或者擴(kuò)展一些功能。比如日志中間件可以方便的記錄請(qǐng)求和響應(yīng)日志,而不需要去更改邏輯代碼。

那么我們簡(jiǎn)化一下軟件執(zhí)行過程,現(xiàn)在有一個(gè)核心類kernel,下面是它的laravel代碼

#捕獲請(qǐng)求
$request = Illuminate\Http\Request::capture()
#處理請(qǐng)求
$response = $kernel->handle($request);

代碼的作用是 捕獲一個(gè) Request ,返回一個(gè) Response。這里面就是后續(xù)分發(fā)到具體執(zhí)行邏輯的代碼段并返回結(jié)果。

那么如果想在執(zhí)行這個(gè)$kernel->handle()方法之前或者之后,增加一段邏輯一般會(huì)怎么寫呢。大概如下:

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

顯然這樣寫沒有問題,但是毫無拓展性可言,想執(zhí)行什么東西都要更改這個(gè)方法,這種是不可能封裝成框架核心內(nèi)容的。怎么改進(jìn)呢

定義一個(gè)要執(zhí)行的中間件類叫middleware,類實(shí)現(xiàn)兩個(gè)方法,before()和after()然后代碼如下。

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

是否解決了問題呢,是解決了不用更改的問題,但是我們?nèi)绻枰鄠€(gè)中間件怎么辦呢,最容易想到的就是:定義一個(gè)中間件數(shù)組middleware_arr,每一個(gè)middleware類都含有before和after方法,代碼如下:

配置項(xiàng)中有middleware_arr
middleware_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()
    }
}

雖然有點(diǎn)老土,但是的確解決了問題。但是這個(gè)還存在一個(gè)問題,就是我們?cè)趺聪蛑虚g件傳遞參數(shù)的問題,那么如下可以嗎:

$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)
    }
}

看似是解決了問題,但是仔細(xì)分析,就會(huì)發(fā)現(xiàn),這里面每次給中間件的都是最初的$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)
    }
}

還有一個(gè)問題就是,假設(shè)有兩個(gè)中間件A和B,那么執(zhí)行順序應(yīng)該是怎么樣呢:

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

這樣合理嗎?不太好分辨,我們假設(shè)有一個(gè)記錄請(qǐng)求和響應(yīng)日志的中間件,這個(gè)時(shí)候,不論你把它放在什么位置,都不能完美的記錄最初請(qǐng)求和最終日志。難道類似情況要寫兩個(gè)類,一個(gè)記錄請(qǐng)求放在中間件數(shù)組第一個(gè),一個(gè)處理響應(yīng),放在數(shù)組最后一位嗎?不如在執(zhí)行后面的foreach之前把middleware_arr數(shù)組給反轉(zhuǎn)一下,這樣就符合了要求:

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

但是我也開始懷疑這個(gè)老土且不靈活的方案是否有更好的解決辦法,在觀察這個(gè)執(zhí)行順序的時(shí)候,發(fā)現(xiàn)是一個(gè)包裹樣式(洋蔥式)的。那個(gè)接下來的問題就能不能找到更靈活精美的解決方案,看上面這種結(jié)構(gòu),總感覺有點(diǎn)熟悉,他很像是A的函數(shù)包裹B的函數(shù),B的函數(shù)包括了最初的執(zhí)行代碼。函數(shù)內(nèi)部調(diào)用函數(shù)容易,但是咱們這里每一個(gè)中間件之間是不知道對(duì)方存在的,所以要把其他中間件要執(zhí)行的函數(shù)傳遞到上一級(jí),這里就用到了閉包函數(shù)還有一個(gè)php函數(shù)array_reduce(),

array_reduce函數(shù)定義:mixed array_reduce ( array $input , callable $function [, mixed $initial = NULL ] )

<?php
function  rsum ( $v ,  $w ){
    $v  +=  $w ;
    return  $v ;
}
function  rmul ( $v ,  $w ){
    $v  *=  $w ;
    return  $v ;
}
$a  = array( 1 ,  2 ,  3 ,  4 ,  5 );
$x  = array();
$b  =  array_reduce ( $a ,  "rsum" );
$c  =  array_reduce ( $a ,  "rmul" ,  10 );
?>  

輸出:

這將使 $b  的值為 15, $c  的值為 1200(= 10*1*2*3*4*5)

array_reduce() 將回調(diào)函數(shù) function 迭代地作用到 input 數(shù)組中的每一個(gè)單元中,從而將數(shù)組簡(jiǎn)化為單一的值。咱們是把多個(gè)函數(shù)包裹成最終調(diào)用一個(gè)函數(shù)。

#我們先假設(shè)只有一個(gè)middleware,叫l(wèi)og來簡(jiǎn)化情況,這里的類應(yīng)該是一個(gè)類全路徑,我這里就簡(jiǎn)單的寫一下,要不然太長(zhǎng)了。
$middleware_arr = ['log'];
#最終要執(zhí)行的代碼先封裝成一個(gè)閉包,要不然沒有辦法傳遞到內(nèi)層,如果用函數(shù)名傳遞函數(shù)的話,是沒有辦法傳遞參數(shù)的。
$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最終是 這樣一個(gè)函數(shù):
function() use($default,$log){
    return $log::handle($default);
};
#所以每一個(gè)中間件都需要有一個(gè)方法handle方法,方法中要對(duì)傳輸?shù)暮瘮?shù)進(jìn)行運(yùn)行,類似如下,這里我類名就不大寫了
class log implements Milldeware {
    public static function handle(Closure $func){
        $func();
    }
}
#這里不難看出可以加入中間件自身邏輯如下:
class log implements Milldeware {
    public static function handle(Closure $func){
        #這里可以運(yùn)行邏輯塊before()
        $func();
        #這里可以運(yùn)行邏輯塊after()
    }
}

這樣在執(zhí)行callback函數(shù)的時(shí)候,執(zhí)行順序如下:

先運(yùn)行l(wèi)og::haddle()方法,

執(zhí)行了log::before()方法

運(yùn)行default方法,執(zhí)行$kernel->handle($request)

運(yùn)行l(wèi)og::after()方法

然后模擬多個(gè)的情況如下:

$middleware_arr = ['csrf','log'];
#最終要執(zhí)行的代碼先封裝成一個(gè)閉包,要不然沒有辦法傳遞到內(nèi)層,如果用函數(shù)名傳遞函數(shù)的話,是沒有辦法傳遞參數(shù)的。
$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最終是 執(zhí)行這樣:
$log::handle(function() use($default,$csrf){
    return $csrf::handle($default);
});

執(zhí)行順序如下:

1.先運(yùn)行l(wèi)og::haddle(包含csrf::handle閉包函數(shù))方法,

2.執(zhí)行了log::before()方法

3.運(yùn)行閉包也就是運(yùn)行了$csrf::handle($default)

4.執(zhí)行了csrf::before()方法

5.運(yùn)行default方法,執(zhí)行$kernel->handle($request)

6.執(zhí)行了csrf::after()方法

7.運(yùn)行l(wèi)og::after()方法

注意這里還有一個(gè)問題就是中間件產(chǎn)生的結(jié)果,并沒有進(jìn)行傳遞,可以通過修改共有資源的方式來達(dá)到相同的目的,并非需要真的傳值到下一個(gè)中間件。

到此這篇文件就結(jié)束了,其實(shí)其中很多關(guān)節(jié)都是我寫這篇文章的時(shí)候才想明白的。尤其是對(duì)閉包函數(shù)的運(yùn)用和理解更深了,閉包函數(shù)可以延遲利用資源,比如當(dāng)前不適合執(zhí)行的語句,又要傳遞到后面,利用閉包可以封裝起來傳遞出去,這是傳統(tǒng)函數(shù)做不到的。

以上就是淺談laravel中間件的創(chuàng)建思路的詳細(xì)內(nèi)容,更多關(guān)于laravel中間件的創(chuàng)建思路的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • php實(shí)現(xiàn)的Cookies操作類實(shí)例

    php實(shí)現(xiàn)的Cookies操作類實(shí)例

    這篇文章主要介紹了php實(shí)現(xiàn)的Cookies操作類及其用法實(shí)例,包括了常見了保存、讀取、更新及清除cookie等操作,在需要進(jìn)行cookie操作時(shí)非常具有實(shí)用價(jià)值,需要的朋友可以參考下
    2014-09-09
  • PHP實(shí)現(xiàn)適用于文件內(nèi)容操作的分頁類

    PHP實(shí)現(xiàn)適用于文件內(nèi)容操作的分頁類

    這篇文章主要為大家詳細(xì)介紹了PHP實(shí)現(xiàn)適用于文件內(nèi)容操作的分頁類,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2016-06-06
  • 詳解PHP中array_rand函數(shù)的使用方法

    詳解PHP中array_rand函數(shù)的使用方法

    大家都知道array_rand是數(shù)組隨機(jī)函數(shù)了,今天看到一個(gè)站長(zhǎng)簡(jiǎn)單的介紹array_rand性能了,于是把許久沒寫的php再來簡(jiǎn)單的看看,我們一起來看看array_rand函數(shù)用法吧。有需要的朋友們可以參考學(xué)習(xí)。
    2016-09-09
  • PHP類繼承 extends使用介紹

    PHP類繼承 extends使用介紹

    我們?cè)谖恼轮袨榇蠹以敿?xì)介紹了有關(guān)PHP類繼承 extends的實(shí)現(xiàn)方法,希望對(duì)于新手來說能夠提升他們?cè)赑HP語言編程中的能力
    2014-01-01
  • php7性能提升的原因詳解

    php7性能提升的原因詳解

    在本篇文章里小編給大家分享是的關(guān)于php7性能提升的原因以及相關(guān)知識(shí)點(diǎn),有需要的朋友們參考下。
    2019-10-10
  • Optimizer與Debugger兼容性問題的解決方法

    Optimizer與Debugger兼容性問題的解決方法

    網(wǎng)上許多聲音說Optimizer與Debugger有沖突,不能同時(shí)開。其實(shí)是可以的,他們兩個(gè)都是Zend擴(kuò)展插件,兩個(gè)單獨(dú)安裝都可以使用,但是要同時(shí)安裝的話必須使用一個(gè)管理器
    2008-12-12
  • PHP+Ajax實(shí)現(xiàn)的無刷新分頁功能詳解【附demo源碼下載】

    PHP+Ajax實(shí)現(xiàn)的無刷新分頁功能詳解【附demo源碼下載】

    這篇文章主要介紹了PHP+Ajax實(shí)現(xiàn)的無刷新分頁功能,結(jié)合具體實(shí)例形式分析了php+ajax通過pdo操作類讀取數(shù)據(jù)庫并分頁顯示的相關(guān)實(shí)現(xiàn)技巧,并附帶demo源碼供讀者下載參考,需要的朋友可以參考下
    2017-07-07
  • php獲取服務(wù)器端mac和客戶端mac的地址支持WIN/LINUX

    php獲取服務(wù)器端mac和客戶端mac的地址支持WIN/LINUX

    這篇文章主要介紹了php獲取服務(wù)器端mac和客戶端mac地址的方法,需要的朋友可以參考下
    2014-05-05
  • php中獲取主機(jī)名、協(xié)議及IP地址的方法

    php中獲取主機(jī)名、協(xié)議及IP地址的方法

    這篇文章主要介紹了php中獲取主機(jī)名、協(xié)議及IP地址的方法,其中涉及gethostbyaddr、gethostbynamel、getprotobyname、getprotobynumber等函數(shù)及$_SERVER變量的應(yīng)用,需要的朋友可以參考下
    2014-11-11
  • php獲取twitter最新消息的方法

    php獲取twitter最新消息的方法

    這篇文章主要介紹了php獲取twitter最新消息的方法,涉及php操作curl及正則替換的技巧,非常具有實(shí)用價(jià)值,需要的朋友可以參考下
    2015-04-04

最新評(píng)論