Der Controller von laravel5.5 bietet die Möglichkeit, automatisch entsprechend dem Methodenparametertyp zu injizieren. Manchmal ist dies jedoch etwas unpraktisch, was sich darin widerspiegelt, dass die Injektion von Methodenparametern nicht vollständig auf dem Parameternamen basiert. Wenn die Reihenfolge der eingehenden Parameter geändert wird, führt dies zu einem Typkonfliktfehler. In diesem Artikel wird das Problem anhand einer eingehenden Analyse des Injektionsprinzips gelöst.
1. Design der Controller-Methodenparameter-Injektionsschritte
1. Fügen Sie Routen in /routes/web.php hinzu
Route::get('/diary/show/{diary}/{page?}','Diary\DiaryController@list');
2. Schreiben Sie die Controller-Datei php unter dem Pfad /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. Erstellen Sie das Modell AppDiary und installieren Sie es in der Datenbank (weggelassen)
<?php namespace App; use Illuminate\Database\Eloquent\Model; class Diary extends Model { protected $table='diary'; public $timestamps = false; }
4 Controller-Methode
Öffnen Sie den Browser und geben Sie ein: „http://127.0.0.1//diary/show/4/12“
Zu diesem Zeitpunkt sind der Titelfeldwert und die ID=4 in der Datentabelle Tagebuch werden ausgegeben.
2. Beschreibung des Injektionsparametertyps
Hinweis: Laravel generiert eine Instanz basierend auf den passenden {diary}- und {page}-Variablen in der Anforderungsroute und die in der Controller-Methode erforderlichen Methodenparametertypen.
Es gibt drei Situationen für verschiedene Parametertypen:
1 Schnittstelle (d. h. von IlluminateDatabaseEloquentModel geerbt), durchsuchen Sie dann im Modell die Tabelle, die dem Objekt entspricht, nach dem Datensatz, dessen ID-Wert der übereinstimmende Parameterwert in der Route ist, und erstellen Sie das Modellobjekt
2. Wenn der Parametertyp ein benutzerdefinierter Typ ist (die UrlRoutable-Schnittstelle ist nicht implementiert), erstellt Laravel Inject nach einem Objekt.
3 Wenn der Parametertyp der Basisdatentyp ist und der Name der in definierte ist Der Routing-Parameter erhält den Wert aus dem Routing-Parameter.
4 Wenn der Parametertyp ein Basisdatentyp ist, der Name jedoch nicht in den Routing-Parametern definiert ist, Verwenden Sie den Standardwert, andernfalls meldet das System einen Fehler.
3. Analyse aktueller Probleme mit injizierten Parametern
In Bezug auf das Spring MVC-Framework von Java weist die Parametertypinjektion von Laravel immer noch Mängel auf, die sich hauptsächlich darin widerspiegeln, dass sie nicht vollständig gemäß dem injiziert wird Parametername.
1. Wenn Sie die Reihenfolge der Controller-Parameter ändern, tritt ein Parametertyp-Übertragungsfehler auf. Wenn Sie die Reihenfolge der Parameter der von DiaryController gesteuerten Show-Methode ändern, tritt ein Fehler auf:
<?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. Da es sich bei dem Parametertyp um einen Basisdatentyp handelt (siehe 2 (3)), handelt es sich nicht um einen dem Namen entsprechend injizierten Parameter. Ändern Sie daher den Code wie folgt, damit er auch normal ausgeführt wird
<?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); } }
4. laravel5. 5 Controller-Methoden-Parameter-Injection-Quellcode-Analyse
1 Der Parametertyp, der die UrlRoutable-Schnittstelle implementiert, wird von der Routing-Middleware IlluminateRoutingMiddlewareSubstituteBinding
public function handle($request, Closure $next) { $this->router->substituteBindings($route = $request->route()); $this->router->substituteImplicitBindings($route); return $next($request); }
public function substituteImplicitBindings($route) { ImplicitRouteBinding::resolveForRoute($this->container, $route); }
public static function resolveForRoute($container, $route) { //从路由参数中获取参数值,$parameters为 ['diary':'4','page':'12'] $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); } }
public function dispatch(Route $route, $controller, $method) { //解析控制器方法的参数 $parameters = $this->resolveClassMethodDependencies( $route->parametersWithoutNulls(), $controller, $method ); if (method_exists($controller, 'callAction')) { //通过Illuminate\Routing\Controller的callAction调用控制器方法 return $controller->callAction($method, $parameters); } //直接调用控制器方法 return $controller->{$method}(...array_values($parameters)); }
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; }
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; }
Verwandte Empfehlungen:
Was ist Abhängigkeitsinjektion?
Wie verwende ich die entsprechende Schnittstelle von Laravel 5.5?
Das obige ist der detaillierte Inhalt vonProblem und Lösung der Laravel5.5-Controller-Parameterübergabereihenfolge. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!