> 백엔드 개발 > PHP 튜토리얼 > PHP DI(종속성 주입) 및 IoC(제어 반전)에 대한 실용적인 튜토리얼을 공유하세요.

PHP DI(종속성 주입) 및 IoC(제어 반전)에 대한 실용적인 튜토리얼을 공유하세요.

零下一度
풀어 주다: 2023-03-10 11:54:01
원래의
1427명이 탐색했습니다.

이 글에서는 주로 PHP 의존성 주입(DI)과 제어 역전(IoC) 관련 정보를 소개하는데, 관심 있는 친구들은 참고해 보세요.

먼저 의존성 주입과 제어 역전에 대해 이야기해 보겠습니다. 마찬가지입니다. 디자인 패턴입니다. 이 디자인 패턴은 프로그램 간의 커플링을 줄이기 위해 사용하는 것입니다. 조금 연구해 본 결과 TP 공식 홈페이지에 관련 기사가 없는 것을 보고 이를 소개하기 위해 겸손하게 글을 썼습니다. 디자인 패턴. TP 커뮤니티에 뭔가 기여할 수 있기를 바랍니다.

우선, 이 디자인 패턴의 정의를 추구하지 마세요. 그렇지 않으면 분명히 구름 속에 있다고 말할 것입니다. 저자는 이에 깊은 영향을 받았는데, 모두 설명되어 있습니다. 이론적 관점에서 볼 때 자바 코드로 설명하는 어휘도 낯설게 가득합니다.

어쨌든 드디어 PHP의 관점에서 종속성 주입의 개념을 설명하겠습니다.

먼저 여기에 데이터베이스 연결을 사용해야 하는 클래스가 있다고 가정합니다. 가장 원시적인 방법에 따르면 이 클래스를 다음과 같이 작성할 수 있습니다. 함수에 먼저 데이터베이스 클래스 파일을 포함합니다.
그런 다음 새 Db를 통해 db 클래스를 인스턴스화하고 데이터베이스 연결 정보를 전달합니다.

그 후 getList 메소드는 $this->_db를 통해 데이터베이스 클래스를 호출할 수 있습니다. 운영.


원하는 기능을 달성한 것 같지만 앞으로는 example1, example2, example3... 다 작성하면 점점 더 많은 클래스가 db 구성요소를 사용해야 할 것입니다. 그렇다면 언젠가 데이터베이스 비밀번호가 변경되거나 db 클래스가 변경되면 다시 돌아가서 모든 클래스 파일을 수정해야 하지 않을까요?

좋아요, 이 문제를 해결하기 위해 팩토리 패턴이 나타났습니다. Factory 메서드를 만들고 Factory::getDb() 메서드를 통해 db 구성 요소의 인스턴스를 얻었습니다.


class example {
  
  private $_db;
  function __construct(){
    include "./Lib/Db.php";
    $this->_db = new Db("localhost","root","123456","test");
  }
  function getList(){
    $this->_db->query("......");//这里具体sql语句就省略不写了
  }
 }
로그인 후 복사

sample 클래스는 다음과 같습니다.


class Factory {
  public static function getDb(){
    include "./Lib/Db.php";
    return new Db("localhost","root","123456","test");
  }
 }
로그인 후 복사

이게 완벽한가요? 앞으로 모든 클래스(example1, example2, example3...)에 대해 다시 생각해보세요. 생성자에서 Factory::getDb()를 통해 Db 인스턴스를 가져와야 합니다. 실제로는 원본에서 Db 클래스와 직접 상호 작용합니다. 커플링은 Factory 클래스와의 커플링이 됩니다. 팩토리 클래스는 데이터베이스 연결 정보를 패키징하는 데만 도움이 됩니다. 비록 데이터베이스 정보가 변경되면 Factory::getDb() 메서드만 수정하면 되지만 어느 날 갑자기 팩토리 메서드가 이름을 변경해야 하거나 getDb 메소드의 이름을 변경해야 하는 경우 어떻게 해야 합니까? 물론 이런 종류의 요구는 실제로 매우 엉망이지만 때로는 이런 상황이 존재합니다. 한 가지 해결책은 다음과 같습니다.

우리는 예제 클래스 내에서 Db 구성 요소를 인스턴스화하지 않습니다. 그 말은? 다음 예제를 보세요.


class example {
  
  private $_db;
  function __construct(){
    $this->_db = Factory::getDb();
  }
  function getList(){
    $this->_db->query("......");//这里具体sql语句就省略不写了
  }
 }
로그인 후 복사

이렇게 하면 예제 클래스가 외부 클래스와 완전히 분리됩니다. Db 클래스에는 팩토리 메서드나 Db 클래스가 없음을 알 수 있습니다. 외부에서 예제 클래스의 setDb 메소드를 호출하여 연결 인스턴스를 직접 주입합니다. 이런 방식으로 예제에서는 db 연결이 어떻게 생성되는지 걱정할 필요가 없습니다.
이를 종속성 주입이라고 합니다. 구현은 코드 내에서 종속성 관계를 생성하지 않지만 이를 매개 변수로 전달합니다. 이렇게 하면 프로그램을 유지 관리하기가 더 쉬워지고 프로그램 코드의 결합 정도가 줄어들며 느슨한 결합이 달성됩니다.

아직 끝나지 않았습니다. db 외에도 다른 외부 클래스가 예제 클래스에 사용된다고 가정해 보겠습니다.


class example {
  private $_db;
  function getList(){
    $this->_db->query("......");//这里具体sql语句就省略不写了
  }
  //从外部注入db连接
  function setDb($connection){
    $this->_db = $connection;
  }
 }
 //调用
$example = new example();
$example->setDb(Factory::getDb());//注入db连接
$example->getList();
로그인 후 복사

너무 많은 세트를 끝없이 작성합니까? 피곤하세요?

좋아요, 매번 너무 많은 코드 줄을 작성하지 않기 위해 또 다른 팩토리 메소드를 만들었습니다:



$example->setDb(Factory::getDb());//注入db连接
$example->setFile(Factory::getFile());//注入文件处理类
$example->setImage(Factory::getImage());//注入Image处理类
 ...
로그인 후 복사

예제를 인스턴스화하면 다음과 같습니다:



class Factory {
  public static function getExample(){
    $example = new example();
    $example->setDb(Factory::getDb());//注入db连接
    $example->setFile(Factory::getFile());//注入文件处理类
    $example->setImage(Factory::getImage());//注入Image处理类
    return $expample;
  }
 }
로그인 후 복사

It 완벽해 보이는데 왜 위의 팩토리 메소드를 처음 사용했을 때의 현장으로 돌아간 것 같은 느낌이 들까요? 이는 실제로 좋은 솔루션이 아니므로 IoC 컨테이너 및 DI 컨테이너라고도 하는 컨테이너라는 또 다른 개념이 제안됩니다.

원래는 setXXX 메소드를 통해 다양한 클래스를 주입했는데, 팩토리 메소드를 통해 패키징할 수도 있지만 그다지 멋지지는 않습니다. 따라서 팩토리 메소드가 필요하지 않습니다. 2. 일단 패키징한 후에 종속성 주입을 어떻게 구현할 수 있습니까?
여기서 규칙을 소개합니다. 다음과 같이 예제 클래스의 생성자에 Di $di라는 매개변수를 전달합니다.


$example=Factory::getExample();
$example->getList();
로그인 후 복사

Di는 소위 IoC 컨테이너로 우리가 사용하는 다양한 항목을 저장하는 것입니다. Type 인스턴스를 사용할 수 있으며, $di->set()을 통해 db라는 인스턴스를 설정합니다. 콜백 함수를 통해 전달되기 때문에 db 클래스는 설정 시 즉시 인스턴스화되지 않고, 설정 시에만 인스턴스화됩니다. $di->get('db')가 실행될 때 마찬가지로 싱글톤 모드도 di 클래스의 디자인에 통합될 수 있습니다.

이런 방식으로 전역 범위에서 Di 클래스를 선언하고 주입해야 하는 모든 클래스를 컨테이너에 넣은 다음 컨테이너를 생성자의 매개변수로 예제에 전달하면 됩니다. 예제 클래스의 컨테이너에서 인스턴스를 가져올 수 있습니다. 물론 생성자일 필요는 없습니다. setDi(Di $di) 메소드를 사용하여 Di 컨테이너를 전달할 수도 있습니다. 간단히 말해서 계약은 사용자가 작성하므로 직접 알아두면 됩니다. .

이제 종속성 주입과 핵심 컨테이너 개념이 소개되었으므로 이제 실제로 사용하고 이해하는 일만 남았습니다!

위 내용은 PHP DI(종속성 주입) 및 IoC(제어 반전)에 대한 실용적인 튜토리얼을 공유하세요.의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

원천:php.cn
본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.
인기 튜토리얼
더>
최신 다운로드
더>
웹 효과
웹사이트 소스 코드
웹사이트 자료
프론트엔드 템플릿