우리 모두 알고 있듯이, 최근에는 그 우아함으로 유명한 프레임워크가 점차 알려지면서 국내 PHP 사용자들에게도 사용되기 시작했습니다. 그러나 Larave는 문서 내용이 형편없다는 명백한 단점을 가지고 있습니다. 본 글은 주로 소스코드를 통해 Laravel의 의존성 주입을 파싱하는 방법에 대한 관련 정보를 소개하고 있으며, 샘플 코드를 통해 매우 자세하게 소개하고 있어 모든 분들의 학습이나 업무에 도움이 되기를 바랍니다.
Laravel 컨트롤러의 생성자 또는 멤버 메소드에서 다음과 같은 유형 제약을 통해 종속성 주입을 사용할 수 있습니다.
public function store(Request $request) { //TODO }
여기서 $request 매개변수는 유형 제약을 사용합니다. 요청은 클래스입니다: IlluminateHttpRequest는 매개변수가 다음과 같아야 함을 의미합니다. 이 클래스 또는 하위 클래스.
이 기사에서는 Laravel의 소스 코드를 분석하여 메소드에 인스턴스를 전달하지 않고 Request를 직접 사용할 수 있는 이유를 알아봅니다. 단지 프레임워크가 자동으로 매개변수를 인스턴스화하고 전달하는 것뿐입니다.
1. 경로 정의
소스에서 시작하여 이러한 경로는 경로 정의 파일에 정의됩니다.
Route::resource('/role', 'Admin\RoleController');
이것은 리소스 기반 경로이며 Laravel은 추가, 삭제, 수정 및 쿼리를 위한 경로 항목을 자동으로 생성합니다. .
이 글의 시작 부분에 있는 store 메소드는 컨트롤러 메소드입니다. 경로에 정의된 Action은 그림에서 볼 수 있습니다: AppHttpControllersAdminRoleController@store
Route 메소드 분석
에 따라 컨트롤러와 메소드를 찾으세요. 경로를 정의하고 디스패치 메서드에 구현됨에서 특정 메서드를 실행합니다.
(파일:vendor/laravel/framework/src/Illuminate/Routing/ControllerDispatcher.php)
public function dispatch(Route $route, $controller, $method) { $parameters = $this->resolveClassMethodDependencies( $route->parametersWithoutNulls(), $controller, $method ); if (method_exists($controller, 'callAction')) { return $controller->callAction($method, $parameters); } return $controller->{$method}(...array_values($parameters)); }
첫번째 해결ClassMethodDependities 메소드, "이름에서 알 수 있듯이"는 클래스의 메소드 매개변수에 따라 종속 객체를 얻는 것입니다. 그런 다음 클래스 메소드를 호출하고 객체 매개변수 주입을 넣습니다.
종속 개체가 여러 개인 경우 foreach에 의해 차례로 구문 분석되어 매개 변수로 주입됩니다.
종속 개체에 대한 코드 가져오기:
protected function resolveClassMethodDependencies(array $parameters, $instance, $method) { if (! method_exists($instance, $method)) { return $parameters; } return $this->resolveMethodDependencies( $parameters, new ReflectionMethod($instance, $method) ); }
여기서 핵심은 PHP 리플렉션을 사용하는 것입니다. RelectionMethod 메서드는 클래스의 메서드 매개 변수 목록을 가져오고 매개 변수 유형 제약 조건, 매개 변수를 알 수 있습니다. 이름 등
여기서 $instance 매개변수는 RoleController 컨트롤러 클래스이고, $method 매개변수는 메소드 이름 store입니다.
2. 종속 개체를 가져오는 예
의 매개 변수에서 종속 개체의 제약 유형을 가져온 후 메서드를 사용하면 이 종속 개체를 인스턴스화할 수 있습니다.
protected function transformDependency(ReflectionParameter $parameter, $parameters) { $class = $parameter->getClass(); // If the parameter has a type-hinted class, we will check to see if it is already in // the list of parameters. If it is we will just skip it as it is probably a model // binding and we do not want to mess with those; otherwise, we resolve it here. if ($class && ! $this->alreadyInParameters($class->name, $parameters)) { return $parameter->isDefaultValueAvailable() ? $parameter->getDefaultValue() : $this->container->make($class->name); } }
클래스 이름을 기준으로 컨테이너에서 개체를 가져옵니다. 개체 인스턴스를 바인딩하는 프로세스는 먼저 서비스 공급자에서 정의됩니다.
그런 다음 인스턴스화된 개체를 store 메서드에 전달하면 종속 개체를 사용할 수 있습니다.
3. PHP 리플렉션에 대하여
ReflectionMethod 사용 예를 들어보세요.
class Demo { private $request; public function store(Request $request) { } }
그림과 같이 새로운 ReflectionMethod(Demo::class, 'store')의 내용을 인쇄합니다.
이 메소드의 매개변수 목록과 매개변수의 제약 조건 유형을 얻을 수 있습니다. typeHint, IlluminateHttpRequest.
클래스에 따르면 이름은 서비스 제공자를 통해 초기에 인스턴스를 바인딩하여 컨테이너에서 얻을 수 있습니다.
관련 권장사항:
Lumen 프레임워크의 사용자 정의 종속성 주입에 대한 간략한 토론
리플렉션 메커니즘을 기반으로 자동 종속성 주입을 구현하는 PHP 방법에 대한 자세한 설명_php 기술
위 내용은 Laravel의 종속성 주입을 구문 분석하는 예의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!