이 기사에서는 주로 PHP 제어 반전 및 종속성 주입 예제를 공유합니다. 종속성 주입의 목적은 더 나은 테스트, 관리 및 코드 확장을 위해 느슨하게 결합된 소프트웨어 아키텍처를 달성하는 것입니다.
Inversion of Control: 호출자가 호출 수신자의 도움이 필요한 경우 기존 프로그래밍 프로세스에서는 호출자가 일반적으로 호출 수신자의 인스턴스를 생성하지만 여기서는 호출 수신자의 작업을 더 이상 호출자가 수행하지 않습니다. 그러나 호출 수신자 생성이 호출자 외부로 이동하여 호출 수신자 생성이 반전되고 호출 수신자 생성에 대한 호출자의 제어가 제거되므로 제어 반전이라고 합니다.
종속성 주입: 제어 반전을 달성하기 위한 일반적인 해결책은 호출 수신자 인스턴스를 생성하는 작업을 IoC 컨테이너에 맡기고 생성자/메서드 주입 구현을 통해 호출자에 호출 수신자를 주입하는 것입니다. 호출자와 호출 수신자의 분리를 달성하는 것을 종속성 주입이라고 합니다. 종속성 주입은 제어 역전의 구현입니다. 일반적인 주입 방법에는 세터 주입, 생성자 주입, 속성 주입 등 세 가지가 있습니다.
Container: 객체 생성, 자원 획득, 파괴 등의 수명 주기를 관리하고 객체 간의 종속성을 설정하며 객체 로딩을 지연시킬 수 있습니다. 더 유명한 것은 PHP-DI와 Pimple입니다.
코드 데모 IoC:
애플리케이션에 저장 공간 요구 사항이 있다고 가정할 때 하위 수준 모듈 API가 상위 수준 애플리케이션에서 직접 호출되면 해당 애플리케이션은 하위 수준 모듈에 종속됩니다.
<?php /** * 高层 */ class App { private $writer; public function __construct() { $this->writer = new FloppyWriter(); } public function save() { $this->writer->saveToFloppy(); } } /** * 低层,软盘存储 */ class FloppyWriter { public function saveToFloppy() { echo __METHOD__; } } $app = new App(); $app->save(); // FloppyWriter::saveToFloppy
프로그램을 다른 플랫폼으로 포팅하려고 하는데 해당 플랫폼이 USB 디스크를 저장 매체로 사용한다면 프로그램을 직접 재사용할 수 없고 수정해야 합니다. 이런 경우에는 하위 레벨의 변화로 인해 상위 레벨의 변화도 따르게 되는데, 이는 나쁜 디자인이다. 프로그램은 구체적인 구현에 의존하지 말고 추상적인 인터페이스에 의존해야 합니다. 코드 데모를 참조하세요.
<?php /** * 接口 */ interface IDeviceWriter { public function saveToDevice(); } /** * 高层 */ class App { /** * @var IDeviceWriter */ private $writer; /** * @param IDeviceWriter $writer */ public function setWriter($writer) { $this->writer = $writer; } public function save() { $this->writer->saveToDevice(); } } /** * 低层,软盘存储 */ class FloppyWriter implements IDeviceWriter { public function saveToDevice() { echo __METHOD__; } } /** * 低层,USB盘存储 */ class UsbDiskWriter implements IDeviceWriter { public function saveToDevice() { echo __METHOD__; } } $app = new App(); $app->setWriter(new UsbDiskWriter()); $app->save(); // UsbDiskWriter::saveToDevice $app->setWriter(new FloppyWriter()); $app->save(); // FloppyWriter::saveToDevice
제어가 실제 FloppyWriter에서 추상 IDeviceWriter 인터페이스로 전송되어 앱이 IDeviceWriter 인터페이스에 종속되고 FloppyWriter 및 UsbDiskWriter도 IDeviceWriter 인터페이스에 종속됩니다.
이것이 IoC입니다. 변경 사항에 직면하여 상위 수준은 코드 한 줄을 수정할 필요가 없으며 더 이상 하위 수준에 의존하지 않고 주입에 의존하여 DI로 이어집니다.
이 구성 요소에 종속성이 많은 경우 종속성을 전달하기 위해 여러 매개 변수 설정 메서드를 만들어야 하므로 코드를 유지 관리하기가 어렵습니다.
<?php //创建依赖实例 $request = new Request(); $filter = new Filter(); //把实例作为参数传递给构造函数 $some = new SomeComponent($request, $filter); $some->setRequest($request); $some->setFilter($filter);
해결책은 종속 인스턴스에 대한 컨테이너를 제공하는 것입니다. 이 컨테이너는 특정 인스턴스가 아닌 컨테이너에 주입되는 전역 레지스트리 역할을 합니다.
<?php class SomeComponent { protected $_di; public function __construct($di) { $this->_di = $di; } public function someRequest() { // 请求实例 $connection = $this->_di->get('request'); } public function someOtherRequest() { // 请求实例 $connection = $this->_di->get('request'); // 过滤器实例 $filter = $this->_di->get('filter'); } } $di = new DI(); //在容器中注册一个request服务 $di->set('request', function() { return new Request(array( "test" => "test" )); }); //在容器中注册一个filter服务 $di->set('filter', function() { return new Filter(); }); //把传递服务的容器作为唯一参数传递给组件 $some = new SomeComponent($di); $some->someRequest();
이 구성 요소는 이제 필요한 서비스를 쉽게 얻을 수 있습니다. 서비스는 지연 로딩을 사용하고 필요할 때만 초기화되므로 서버 리소스도 절약됩니다. 이 구성 요소는 이제 크게 분리되었습니다.
관련 권장사항:
PHP 종속성 주입(DI) 및 제어 역전(IoC) 예제 자습서
위 내용은 PHP 제어 반전 및 종속성 주입 예제의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!