이전 기사에서는 종속성 주입을 설명하기 위해 특정 웹 사례를 사용했습니다. 오늘은 종속성 주입 컨테이너(컨테이너)에 대해 먼저 설명하겠습니다.
Big Most of 종속성 주입을 사용하여 구성 요소를 분리하는 시점에는 컨테이너가 필요하지 않습니다.
그러나 다양한 개체를 관리하고 개체 간의 복잡한 종속성을 처리하려는 경우 컨테이너가 매우 유용합니다.
첫 번째 기사의 예를 기억하시나요? User 개체를 만들기 전에 먼저 SessionStorage 개체를 만들어야 합니다. 별것 아니지만, 그래도 필요한 객체를 만들기 전에 그것이 의존하는 객체를 명확히 알기 때문이라고 말하고 싶습니다. 객체가 많고 종속성이 복잡하면(SessionStorage 클래스는 캐시 클래스에 종속되고 캐시 클래스는 파일 클래스와 inputFilter 클래스에 종속되며 파일 클래스는 stdio 클래스에 종속된다고 가정) 문제가 됩니다. . . . .
$storage = new SessionStorage('SESSION_ID'); $user = new User($storage);
다음 기사에서는 Symfony 2에서 컨테이너를 구현하는 방법을 소개하겠습니다. 그러나 지금은 컨테이너를 간단하고 명확하게 설명하기 위해 Symfony를 무시하겠습니다. 다음은 Zend Framework의 예를 사용하여 설명합니다.
Zend Framework의 Zend_Mail 클래스는 이메일 관리를 단순화합니다. 기본적으로 PHP의 mail() 기능을 사용하여 이메일을 보내지만 유연성이 좋지 않습니다. 다행히도 이 동작은 전송 클래스를 제공하여 쉽게 변경할 수 있습니다.
다음 코드는 Zend_Mail 클래스를 생성하고 Gmail 계정을 사용하여 이메일을 보내는 방법을 보여줍니다.
$transport = new Zend_Mail_Transport_Smtp('smtp.gmail.com', array( 'auth' => 'login', 'username' => 'foo', 'password' => 'bar', 'ssl' => 'ssl', 'port' => 465, )); $mailer = new Zend_Mail(); $mailer->setDefaultTransport($transport);
종속성 주입 컨테이너는 관리하는 다양한 항목을 인스턴스화하고 구성할 수 있는 대규모 클래스입니다. 수업. 이를 수행하려면 해당 클래스의 생성자 메서드의 매개 변수와 종속성을 알아야 합니다.
다음은 이전에 언급한 Zend_Mail 객체를 얻는 작업을 계속 구현하는 하드 코딩된 컨테이너입니다.
class Container { public function getMailTransport() { return new Zend_Mail_Transport_Smtp('smtp.gmail.com', array( 'auth' => 'login', 'username' => 'foo', 'password' => 'bar', 'ssl' => 'ssl', 'port' => 465, )); } public function getMailer() { $mailer = new Zend_Mail(); $mailer->setDefaultTransport($this->getMailTransport()); return $mailer; } } //容器的使用也很简单 $container = new Container(); $mailer = $container->getMailer();
컨테이너를 사용할 때 Zend_Mail 객체를 얻어야 하는 경우에는 다음을 수행하지 마세요. 객체 인스턴스 생성에 대한 모든 세부 사항은 컨테이너에 내장되어 있으므로 생성 세부 사항을 알아야 합니다. Mail_Transport 클래스에 대한 Zend_Mail의 종속성은 컨테이너를 통해 Zend_Mail 개체에 자동으로 주입될 수도 있습니다.
종속 객체 획득은 주로 getMailTransport()에 의해 구현됩니다. 컨테이너의 성능은 이 간단한 get 호출을 통해 달성됩니다.
하지만 똑똑하다면 컨테이너에 하드코딩(예: 이메일 전송을 위한 계정 및 비밀번호 정보 등)이 있다는 것을 발견했을 것입니다. 따라서 우리는 한 단계 더 나아가 컨테이너에 매개변수를 추가하여 컨테이너를 더욱 유용하게 만들어야 합니다.
class Container { protected $parameters = array(); public function __construct(array $parameters = array()) { $this->parameters = $parameters; } public function getMailTransport() { return new Zend_Mail_Transport_Smtp('smtp.gmail.com', array( 'auth' => 'login', 'username' => $this->parameters['mailer.username'], 'password' => $this->parameters['mailer.password'], 'ssl' => 'ssl', 'port' => 465, )); } public function getMailer() { $mailer = new Zend_Mail(); $mailer->setDefaultTransport($this->getMailTransport()); return $mailer; } }
이제 컨테이너 생성자의 매개변수를 통해 이메일 전송을 위한 계정과 비밀번호를 쉽게 전환할 수 있습니다.
$container = new Container(array( 'mailer.username' => 'foo', 'mailer.password' => 'bar', )); $mailer = $container->getMailer();
Zend_Mail 클래스가 현재 요구 사항을 충족할 수 없다고 생각되면 테스트할 때 일부 로그가 필요함), 메일 전송 클래스를 쉽게 전환하려면 컨테이너 생성자의 매개변수를 통해 클래스 이름을 전달할 수도 있습니다
class Container { // ... public function getMailer() { $class = $this->parameters['mailer.class']; $mailer = new $class(); $mailer->setDefaultTransport($this->getMailTransport()); return $mailer; } } $container = new Container(array( 'mailer.username' => 'foo', 'mailer.password' => 'bar', 'mailer.class' => 'MyTest_Mail', )); $mailer = $container->getMailer();
마지막으로 고려사항 고객이 메일러 개체(오버헤드)를 얻을 때마다 다시 인스턴스화할 필요가 없으므로 컨테이너는 매번 동일한 개체 인스턴스를 제공해야 합니다.
따라서 프로그램은 보호된 정적 배열 $shared를 사용하여 첫 번째 인스턴스화된 개체를 저장합니다. 나중에 사용자가 메일러를 얻을 때 첫 번째 인스턴스화된 개체가 반환됩니다.
class Container { static protected $shared = array(); // ... public function getMailer() { if (isset(self::$shared['mailer'])) { return self::$shared['mailer']; } $class = $this->parameters['mailer.class']; $mailer = new $class(); $mailer->setDefaultTransport($this->getMailTransport()); return self::$shared['mailer'] = $mailer; } }
컨테이너 캡슐화 컨테이너가 관리해야 하는 기본 기능에는 개체의 인스턴스화 및 구성이 포함됩니다. 이러한 개체 자체는 컨테이너에 의해 관리된다는 사실을 모르고 컨테이너의 존재를 무시할 수 있습니다. 이것이 컨테이너가 모든 PHP 클래스를 관리할 수 있는 이유입니다. 객체 자체가 종속성을 처리하기 위해 종속성 주입을 사용하면 더 좋지만 물론 그럴 필요는 없습니다.
하지만 손으로 컨테이너를 만들고 유지하는 것은 금방 악몽이 될 수 있습니다. 다음 기사에서는 Symfony 2가 컨테이너를 구현하는 방법을 설명합니다.
위 내용은 PHP 의존성 주입 컨테이너 시리즈(2)를 이해하기 위해 필요한 내용입니다. 더 많은 관련 내용은 PHP 중국어 홈페이지(www.php.cn)를 참고해주세요!