종속성 주입
종속성 주입은 멋진 용어이지만 실제로는 클래스의 종속성이 생성자를 통해 또는 경우에 따라 "setter" 메서드를 통해 "주입"된다는 의미입니다. 먼저 Laravel 컨트롤러의 코드 예제를 살펴보겠습니다.
<?php namespace App\Http\Controllers; use Illuminate\Routing\Controller; use App\Users\Repository as UserRepository; class UserController extends Controller { /** * 用户 Repository 的实例。 */ protected $users; /** * 创建一个新的控制器实例。 * * @param UserRepository $users * @return void */ public function __construct(UserRepository $users) { $this->users = $users; } /** * 显示指定 ID 的用户。 * * @param int $id * @return View */ public function show($id) { $user_info = $this->users->find($id); return view('user', ['user_info' => $user_info]); } }
Laravel은 클래스 종속성을 관리하고 서비스 컨테이너를 통해 종속성 주입을 수행합니다. 함수 매개변수에 대한 유형 힌트로 인터페이스를 사용하는 경우 지정된 구현을 인터페이스에 바인딩해야 합니다.
interface EventPusher { public function send($data); }
class RedisEventPusher implements EventPusher { public function send($data) { // } }
$this->app->bind('App\Contracts\EventPusher', 'App\Services\RedisEventPusher');
use App\Contracts\EventPusher; /** * 创建一个新的类实例。 * * @param EventPusher $pusher * @return void */ public function __construct(EventPusher $pusher) { $this->pusher = $pusher; }
이것이 소위 인터페이스입니다. 지향 프로그래밍에서 인터페이스는 사양과 제약으로 이해될 수 있습니다. 상위 수준 모듈은 하위 수준 모듈에 직접적으로 의존하지 않으며 모두 추상화(인터페이스라고 함)에 의존해야 합니다.
종속성 주입 사용의 가장 중요한 이점은 필요한 객체와 외부 리소스를 효과적으로 분리하여 느슨하게 결합하여 기능 재사용에 도움이 된다는 점이며, 더 중요한 것은 전체 아키텍처를 개선한다는 것입니다. 프로그램이 매우 유연해집니다.
제어 반전
제어 반전(IoC로 약칭)은 객체 지향 프로그래밍의 설계 원칙입니다. 가장 일반적인 방법은 DI(종속성 주입)이고 다른 방법은 "종속성 조회"입니다. 제어 역전을 통해 객체가 생성되면 시스템의 모든 객체를 제어하는 외부 엔터티가 자신이 의존하는 객체의 참조를 전달합니다. 의존성이 객체에 주입된다고도 말할 수 있습니다.
<?php /** * 没有IoC/DI的时候,常规的A类使用C类的示例 */ /** * Class c */ class c { public function say() { echo 'hello'; } } /** * Class a */ class a { private $c; public function __construct() { $this->c = new C(); // 实例化创建C类 } public function sayC() { echo $this->c->say(); // 调用C类中的方法 } } $a = new a(); $a->sayC();
아래 그림과 같이 IoC/DI 컨테이너를 사용하면 클래스 A가 더 이상 C를 적극적으로 생성하지 않습니다.
대신 다음 그림과 같이 IoC/DI 컨테이너가 C 인스턴스를 얻을 때까지 수동적으로 기다린 다음 이를 클래스 A에 역으로 주입합니다.
<?php /** * 当有了IoC/DI的容器后,a类依赖c实例注入的示例 */ /** * Class c */ class c { public function say() { echo 'hello'; } } /** * Class a */ class a { private $c; public function setC(C $c) { $this->c = $c; // 实例化创建C类 } public function sayC() { echo $this->c->say(); // 调用C类中的方法 } } $c = new C(); $a = new a(); $a->setC($c); $a->sayC();