首頁 後端開發 php教程 laravel5.5控制器傳參順序問題及解決方案

laravel5.5控制器傳參順序問題及解決方案

May 10, 2018 pm 04:34 PM
控制器 解決方案

laravel5.5的控制器提供了根據方法參數類型,自動注入的能力。但是有時會略有不便,體現在方法參數的注入不完全是按照參數名稱​​進行的,如果改變了傳入參數的順序會導致類型不匹配的錯誤。本文從其註入的原理深度解析進行解決。

一、控制器方法參數注入步驟設計

1、在/routes/web.php中加入路由

Route::get('/diary/show/{diary}/{page?}','Diary\DiaryController@list');
登入後複製

2、寫控制器檔案DiaryController.php放到/app/Http/Controllers/Diary/路徑下面

<?php
namespace App\Http\Controllers\Diary;
use App\Http\Controllers\Controller;
class DiaryController extends Controller
{
    public function show(\App\Diary $diary,$page=11){
        var_dump($diary->title,$page); 
  } 
}
登入後複製

3、建立模型\App\Diary並安裝到資料庫(略)

<?php
namespace App;
use Illuminate\Database\Eloquent\Model;
class Diary extends Model
{
    protected $table=&#39;diary&#39;;
    public $timestamps = false;
}
登入後複製

4、存取控制器方法

開啟瀏覽器輸入:「http://127.0.0.1//diary/show/4/12」

此時輸出資料表diary中id=4的title欄位值和12

二、注入參數類型說明

說明:laravel會根據請求路由中匹配的{diary}和{page}變數和控制器方法中所需的方法參數類型,產生實例物件並註入到控制器方法中,

針對不同的參數類型分成三種情況:

1、如果參數類型實作了UrlRoutable介面(即繼承自Illuminate\Database\Eloquent\Model),則在模型物件對應的表中尋找id值為路由中符合參數值的那筆記錄,並建構模型物件;

2、如果參數類型為自訂類型(沒有實作UrlRoutable介面),則laravel會建構一個物件後注入;

3、如果參數類型為基礎資料類型,且名稱為路由參數中定義的名稱,則從路由參數中取得值;

4、如果參數類型為基礎資料類型,但名稱未在路由參數中定義,如果有預設值則使用預設值,否則系統會提示錯誤。

三、目前註入參數存在的問題分析

參考java的Spring MVC框架,laravel的參數類型注入還存在缺陷,主要體現在不完全是依照參數名稱​​進行注入。

1、如果改變控制器參數的順序,會出現參數類型傳遞錯誤,如將DiaryController控制的show方法的參數改變順序,則會導致錯誤發生:

<?php
namespace App\Http\Controllers\Diary;
use App\Http\Controllers\Controller;
class DiaryController extends Controller
{
    public function show($page,\App\Diary $diary){
        var_dump($diary->title,$page); 
  } 
}
登入後複製

2、由於參數類型為基礎資料型態(參見二(3)),並不是依照名稱來注入的參數,因此將程式碼改為如下,同樣會運作正常

<?php
namespace App\Http\Controllers\Diary;
use App\Http\Controllers\Controller;
class DiaryController extends Controller
{
    public function show(\App\Diary $diary,$pag){
        var_dump($diary->title,$pag); 
  } 
}
登入後複製

四、laravel5.5控制器方法參數注入源碼剖析

1、實現了UrlRoutable接口的參數類型由路由中間件Illuminate\Routing\Middleware\SubstituteBinding實現構建

    public function handle($request, Closure $next)
    {
        $this->router->substituteBindings($route = $request->route());
        $this->router->substituteImplicitBindings($route);
        return $next($request);
    }
登入後複製

Illuminate\Routing\Router的substituteImplicitBindings方法

    public function substituteImplicitBindings($route)
    {
        ImplicitRouteBinding::resolveForRoute($this->container, $route);
    }
登入後複製

在Illuminate\Routing\ImplicitRouteBinding的resolveForRoute方法中實作

    public static function resolveForRoute($container, $route)
    {
        //从路由参数中获取参数值,$parameters为 [&#39;diary&#39;:&#39;4&#39;,&#39;page&#39;:&#39;12&#39;]
        $parameters = $route->parameters();
        //获取控制器的函数参数列表,此处传入UrlRoutable::class,只返回实现UrlRoutable接口的参数
        $signatureParameters = $route->signatureParameters(UrlRoutable::class);
        foreach ($signatureParameters as $parameter) {
            if (! $parameterName = static::getParameterName($parameter->name, $parameters)) {
                continue;
            }
            $parameterValue = $parameters[$parameterName];
            if ($parameterValue instanceof UrlRoutable) {
                continue;
            }
            //构建模型的实例(基础自Illuminate\Database\Eloquent\Model),此处为App\Diary
            $instance = $container->make($parameter->getClass()->name);
            //将参数值绑定到模型,参加Illuminate\Database\Eloquent\Model的resolveRouteBinding方法
            if (! $model = $instance->resolveRouteBinding($parameterValue)) {
                throw (new ModelNotFoundException)->setModel(get_class($instance));
            }
       //根据参数名称注入模型实例
            $route->setParameter($parameterName, $model);
        }
    }
登入後複製

附加說明:

這裡呼叫$route物件(Illuminate\Routing\Route類型)setParameter方法,說明模型參數類型(見二(1))正是透過參數類型和參數名稱同時匹配才注入模型實例

2、其它類型的控制器參數在運行路由控制器時綁定

#關鍵部分在Illuminate\Routing\ControllerDispatcher的dispatch方法中實作:

    public function dispatch(Route $route, $controller, $method)
    {
        //解析控制器方法的参数
        $parameters = $this->resolveClassMethodDependencies(
            $route->parametersWithoutNulls(), $controller, $method
        );
        if (method_exists($controller, &#39;callAction&#39;)) {
            //通过Illuminate\Routing\Controller的callAction调用控制器方法
            return $controller->callAction($method, $parameters);
        }
        //直接调用控制器方法
        return $controller->{$method}(...array_values($parameters));
    }
登入後複製

呼叫resolveClassMethodDependencies方法

  public function resolveMethodDependencies(array $parameters, ReflectionFunctionAbstract $reflector)
    {
        $instanceCount = 0;
        $values = array_values($parameters);
        //通过方法反射获取方法参数
        foreach ($reflector->getParameters() as $key => $parameter) {
        //如果有默认值则返回默认值,如果是自定义方法则构建实例返回
            $instance = $this->transformDependency(
                $parameter, $parameters
            );
            if (! is_null($instance)) {
                $instanceCount++;
                //自定义类型(未实现UrlRoutable接口)的实例注入
                $this->spliceIntoParameters($parameters, $key, $instance);
            } elseif (! isset($values[$key - $instanceCount]) &&
                      $parameter->isDefaultValueAvailable()) {
                //未在路由参数中定义,但有默认值的参数注入
                $this->spliceIntoParameters($parameters, $key, $parameter->getDefaultValue());
            }
        }
        return $parameters;
    }
登入後複製

問題總結說明:

#1、模型參數(見二(見二( 1))和名稱在路由參數中定義的基礎類型(見二(3))必須按照在路由中定義的順序首先傳入控制器方法,否則會出現類型不匹配的錯誤;

# 2、自訂類型(見二(2))和名稱未在路由參數中定義的基礎型別參數(請參閱二(4)),在控制器方法中依出現的順序依序傳入。

五、問題修復

開啟/vendor/laravel/framework/src/Illuminate/Routing/RouteDependencyResolverTrait.php文件,

將resolveMethodDependencies方法修改為下列程式碼

    public function resolveMethodDependencies(array $parameters,ReflectionFunctionAbstract $reflector){
        $methodParameters=[];
        foreach($reflector->getParameters() as $key=>$parameter){
            $name=$parameter->getName();
            $instance=$this->transformDependency($parameter, $parameters);
            if(!is_null($instance)){
                $methodParameters[]=$instance;
            }elseif(!isset($parameters[$name]) && $parameter->isDefaultValueAvailable()){
                $methodParameters[]=$parameter->getDefaultValue();
            }else{
                $methodParameters[]=isset($parameters[$name]) ? $parameters[$name] : null;
            }
        }
        return $methodParameters;
    }
登入後複製

修改之後完全按照名稱和類型注入控制器方法參數,程式碼變得更簡潔,功能也更強大了!

如果參數沒有在路由中定義且未提供預設值,那麼將以null注入。

相關推薦:

PHP基於反射機制實作自動依賴注入的方法詳解

什麼是依賴注入?

Laravel 5.5的可對應介面如何使用?

#

以上是laravel5.5控制器傳參順序問題及解決方案的詳細內容。更多資訊請關注PHP中文網其他相關文章!

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

AI Hentai Generator

AI Hentai Generator

免費產生 AI 無盡。

熱門文章

R.E.P.O.能量晶體解釋及其做什麼(黃色晶體)
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.最佳圖形設置
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.如果您聽不到任何人,如何修復音頻
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
WWE 2K25:如何解鎖Myrise中的所有內容
4 週前 By 尊渡假赌尊渡假赌尊渡假赌

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

針對Win11無法安裝中文語言包的解決方案 針對Win11無法安裝中文語言包的解決方案 Mar 09, 2024 am 09:15 AM

Win11是微軟推出的最新作業系統,相較於先前的版本,Win11在介面設計和使用者體驗上有了很大的提升。然而,一些用戶反映他們在安裝Win11後遇到了無法安裝中文語言套件的問題,這就給他們在系統中使用中文帶來了困擾。本文將針對Win11無法安裝中文語言套件的問題提供一些解決方案,幫助使用者順利使用中文。首先,我們要明白為什麼無法安裝中文語言包。一般來說,Win11

scipy庫安裝失敗的原因及解決方案 scipy庫安裝失敗的原因及解決方案 Feb 22, 2024 pm 06:27 PM

scipy庫安裝失敗的原因及解決方案,需要具體程式碼範例在進行Python科學計算時,scipy是一個非常常用的函式庫,它提供了許多用於數值計算、最佳化、統計和訊號處理的功能。然而,在安裝scipy庫時,有時會遇到一些問題,導致安裝失敗。本文將探討scipy庫安裝失敗的主要原因,並提供對應的解決方案。安裝依賴套件失敗scipy庫依賴一些其他的Python庫,例如nu

解決Oracle字元集修改造成亂碼問題的有效方案 解決Oracle字元集修改造成亂碼問題的有效方案 Mar 03, 2024 am 09:57 AM

標題:解決Oracle字元集修改造成亂碼問題的有效方案在Oracle資料庫中,當字元集被修改後,往往會因為資料中存在不相容的字元而導致亂碼問題的出現​​。為了解決這個問題,我們需要採取一些有效的方案來處理。本文將介紹一些解決Oracle字元集修改引起亂碼問題的具體方案和程式碼範例。一、匯出資料並重新設定字元集首先,我們可以透過使用expdp指令將資料庫中的資料匯出

Oracle NVL函數常見問題及解決方案 Oracle NVL函數常見問題及解決方案 Mar 10, 2024 am 08:42 AM

OracleNVL函數常見問題及解決方案Oracle資料庫是廣泛使用的關係型資料庫系統,在資料處理過程中經常需要處理空值的情況。為了因應空值所帶來的問題,Oracle提供了NVL函數來處理空值。本文將介紹NVL函數的常見問題及解決方案,並提供具體的程式碼範例。問題一:NVL函式用法不當NVL函式的基本語法為:NVL(expr1,default_value)其

使用C++實現機器學習演算法:常見挑戰及解決方案 使用C++實現機器學習演算法:常見挑戰及解決方案 Jun 03, 2024 pm 01:25 PM

C++中機器學習演算法面臨的常見挑戰包括記憶體管理、多執行緒、效能最佳化和可維護性。解決方案包括使用智慧指標、現代線程庫、SIMD指令和第三方庫,並遵循程式碼風格指南和使用自動化工具。實作案例展示如何利用Eigen函式庫實現線性迴歸演算法,有效地管理記憶體和使用高效能矩陣操作。

解決無法正常啟動應用程式錯誤代碼0xc000007b 解決無法正常啟動應用程式錯誤代碼0xc000007b Feb 20, 2024 pm 01:24 PM

無法正常啟動0xc000007b怎麼解決在使用電腦時,我們有時會遇到各種錯誤代碼,其中最常見的之一就是0xc000007b。當我們嘗試運行某些應用程式或遊戲時,突然出現這個錯誤代碼,使我們無法正常啟動。那麼,我們該如何解決這個問題呢?首先,我們要先了解錯誤代碼0xc000007b的意思。這個錯誤代碼通常指示一個或多個關鍵的系統檔案或庫檔案缺失、損壞或不

揭秘解決PyCharm密鑰失效的方法 揭秘解決PyCharm密鑰失效的方法 Feb 23, 2024 pm 10:51 PM

PyCharm是一款功能強大的Python整合開發環境,廣受開發者喜愛。然而,有時候我們在使用PyCharm時可能會遇到金鑰失效的問題,導致無法正常使用軟體。本文將為大家揭秘PyCharm密鑰失效的解決方案,並提供具體的程式碼範例,幫助讀者快速解決這個問題。在開始解決問題之前,我們首先要了解密鑰失效的原因。 PyCharm的金鑰失效通常是由於網路問題或軟體本身

MySQL安裝中文亂碼的常見原因及解決方案 MySQL安裝中文亂碼的常見原因及解決方案 Mar 02, 2024 am 09:00 AM

MySQL安裝中文亂碼的常見原因及解決方案MySQL是一種常用的關係型資料庫管理系統,但在使用過程中可能會遇到中文亂碼的問題,這給開發者和系統管理員帶來了困擾。中文亂碼問題的出現​​主要是因為字元集設定不正確、資料庫伺服器和客戶端字元集不一致等原因導致的。本文將詳細介紹MySQL安裝中文亂碼的常見原因及解決方案,幫助大家更能解決這個問題。一、常見原因:字元集設

See all articles