이번 시리즈에서는 PHP 객체지향 프로그래밍(OOP)의 기본을 다루겠습니다. 콘텐츠는 순차적인 부분으로 구성되며 각 부분은 특정 주제에 중점을 둡니다. OOP 개념이 처음이거나 익숙하지 않은 경우 이 시리즈를 통해 단계별로 안내를 받을 수 있습니다. 이번 편에서는 PHP의 구성과 상속, 의존성 주입에 대해 살펴보겠습니다. PHP OOP를 배우는 여정을 함께 시작해보세요!
우리는 객체 지향 프로그래밍에서 부모 클래스와 자식 클래스 사이의 관계에 대해 이미 배웠습니다. 여기서 자식 클래스가 부모 클래스를 상속하고 그 클래스의 모든 것에 액세스할 수 있다는 것을 보았습니다. 이를 상속이라고 합니다.
반면, 컴포지션(Composition)은 부모 클래스를 상속받는 것이 아니라 자식 클래스에 속성값으로 할당하는 것을 말합니다. 이를 통해 부모 클래스의 모든 것에 접근할 수 있습니다. 이것을 작곡이라고 합니다.
아래는 구성과 상속을 보여주는 예시입니다.
class Link { public string $name; public string $type; public function create($name, $type) { $this->name = $name; $this->type = $type; } public function show() { echo "name: $this->name, type: $this->type"; } } // Inheritance example class ShoLink extends Link { // other functionalities } // Composition example class User { public Link $link; public function __construct() { $this->link = new Link(); } // other functionalities } $user = new User(); $user->link->create("Jamir", "Short");
첫 번째 예에서는 ShoLink 클래스가 Link 클래스를 상속하는 것을 볼 수 있습니다. 반면에 두 번째 예에서는 User 클래스가 Link 클래스를 상속하지 않습니다. 대신 Link 클래스의 인스턴스를 해당 속성 중 하나에 할당합니다. 결과적으로 두 하위 클래스 모두에서 Link 클래스의 모든 항목에 액세스할 수 있습니다.
이제 질문이 생길 수 있습니다. 이미 상속을 사용하여 모든 것에 액세스할 수 있다면 왜 합성을 사용해야 할까요? 결국 구성을 사용하면 추가 속성을 선언하고 구성을 통해 해당 값을 설정해야 합니다. 이것은 추가 작업처럼 보입니다. 그렇다면 구성을 사용하면 어떤 이점이 있습니까?
우리는 상속을 통해 상위 클래스의 모든 항목을 하위 클래스에서 액세스할 수 있다는 것을 알고 있습니다. 결과적으로 부모 클래스의 특정 메서드를 사용하고 싶지 않거나 부모 클래스의 일부 속성이나 메서드가 자식 클래스에 필요하지 않은 경우에도 공개 또는 보호 멤버인 경우 자식 클래스에서 액세스할 수 있습니다. .
이 문제를 해결하기 위해 구성이 사용됩니다. 구성을 사용하면 상위 클래스의 필수 부분만 하위 클래스에서 액세스할 수 있도록 만들 수 있습니다. 다른 예를 통해 이를 더욱 명확하게 설명하겠습니다.
Link 클래스를 자세히 살펴보면 show 메소드가 있음을 알 수 있습니다. 이 방법을 사용하면 ShoLink 클래스에서 생성된 링크를 직접 표시할 수 있습니다.
그러나 User 클래스를 통해 누구도 사용자를 위해 생성된 링크를 직접 볼 수 없도록 하려면 어떻게 해야 할까요? 대신 프로필과 함께 사용자의 링크를 표시할 수 있습니다.
이것이 User 클래스에서 Link 클래스를 상속하는 대신 구성을 통해 액세스하는 이유입니다. 결과적으로 User 클래스를 통해서는 누구도 해당 사용자의 링크를 직접 볼 수 없지만, ShoLink 클래스의 링크는 직접 볼 수 있습니다.
이제 우리는 구성에 대해 어느 정도 이해했으며 특정 문제를 해결하기 위해 상속 대신 이를 사용해야 하는 경우도 있습니다. OOP에는 "상속보다 구성을 선호한다"는 원칙이 있는데, 이는 상속보다 구성을 우선시한다는 뜻이다. 즉, 상위 클래스의 모든 항목에 액세스할 필요가 없는 하위 클래스의 경우 항상 상속보다 구성을 선호해야 합니다.
이제 질문이 생깁니다. 구성을 사용할 시기와 상속을 사용할 시기를 어떻게 결정합니까?
이 경우 두 가지 관계 유형을 토대로 결정을 내려야 합니다.
class Link { public string $name; public string $type; public function create($name, $type) { $this->name = $name; $this->type = $type; } public function show() { echo "name: $this->name, type: $this->type"; } } // Inheritance example class ShoLink extends Link { // other functionalities } // Composition example class User { public Link $link; public function __construct() { $this->link = new Link(); } // other functionalities } $user = new User(); $user->link->create("Jamir", "Short");
위의 ShoLink 클래스의 예를 보면 ShoLink 클래스가 Link 클래스를 상속하고 있음을 알 수 있습니다. 그래서 이들 사이의 관계를 정의한다면 ShoLink는 본질적으로 링크 유형이기 때문에 ShoLink is a Link 관계가 될 것입니다.
// Inheritance example class ShoLink extends Link { // other functionalities }
이제 위 User 클래스의 예를 보면 User 클래스가 Link 클래스와 합성을 사용하고 있음을 알 수 있습니다. 따라서 이들 사이의 관계를 정의한다면 사용자는 링크가 아니기 때문에 관계는 사용자가 링크를 가지고 있지만 사용자는 링크를 가질 수도 있고 소유할 수도 있습니다.
이제 각각을 언제 사용해야 하는지, 다양한 상황에서 어떤 우선 순위를 두어야 하는지 등 구성과 상속에 대해 더 명확하게 이해하셨기를 바랍니다.
의존성 주입을 이해하기 전에 먼저 종속성이 무엇인지 이해해야 합니다. 종속성은 자식 클래스가 상속이나 구성을 통해 다른 클래스의 멤버를 사용하는 경우입니다. 이 경우 상위 클래스는 하위 클래스의 종속성이 됩니다.
위의 예에서 상속 대신 합성을 사용할 때 하위 클래스에서 속성을 선언하고 생성자를 통해 상위 클래스의 인스턴스를 해당 속성에 할당해야 한다는 것을 확인했습니다. 따라서 User 클래스를 사용하려면 User 클래스가 Link 클래스에 종속되므로 생성자에서 Link 클래스를 인스턴스화해야 합니다. 즉, Link 클래스는 User 클래스에 대한 종속성입니다. 여기서 문제는 Link 클래스의 인스턴스화 프로세스가 User 클래스 내에서 긴밀하게 결합되어 있다는 것입니다.
문제는 Link 클래스의 인스턴스화가 제한적이고 User 클래스에만 국한된다는 것입니다. 외부에서 Link 클래스 대신 다른 클래스를 User 클래스로 전달하려는 경우 생성자에서 Link 클래스의 인스턴스를 명시적으로 생성하고 이를 Link 속성에 할당하기 때문에 그렇게 할 수 없습니다. 이를 긴밀하게 결합된 종속성이라고 하며, 이는 외부에서 이 종속성을 변경할 수 없음을 의미합니다.
그러나 생성자에서 Link 클래스를 직접 인스턴스화하지 않고 대신 사용자에게 맡기면, 즉 사용자가 User 클래스를 사용할 때 Link 클래스 종속성을 User 클래스에 전달하게 되면 문제는 다음과 같습니다. 해결되었습니다.
아래 코드 예시를 살펴보겠습니다.
class Link { public string $name; public string $type; public function create($name, $type) { $this->name = $name; $this->type = $type; } public function show() { echo "name: $this->name, type: $this->type"; } } // Inheritance example class ShoLink extends Link { // other functionalities } // Composition example class User { public Link $link; public function __construct() { $this->link = new Link(); } // other functionalities } $user = new User(); $user->link->create("Jamir", "Short");
이 예에서는 User 클래스의 생성자에서 Link 클래스를 인스턴스화하는 대신 Link 클래스의 종속성을 외부에서 User 클래스로 전달하는 것을 볼 수 있습니다. 사용자를 통해 User 클래스에 종속성을 전달하는 이러한 프로세스를 종속성 주입이라고 합니다. 즉, Link 클래스의 종속성을 외부에서 주입하거나 푸시하는 것입니다. 이를 느슨하게 결합된 종속성이라고 합니다. 즉, 이 종속성을 외부에서 쉽게 변경할 수 있습니다.
이제 Link 클래스에도 자체 종속성이 있는 경우 User 클래스를 통해 외부에서 해당 종속성을 삽입할 수도 있습니다. 그런 다음 Link 클래스의 인스턴스를 User 클래스에 삽입하면 됩니다. 결과적으로 User 클래스 내에서 Link 클래스의 종속성에 대해 걱정할 필요가 없습니다. 사용자가 외부에서 이를 처리하기 때문입니다.
아래 코드 예시를 살펴보겠습니다.
// Inheritance example class ShoLink extends Link { // other functionalities }
이렇게 하면 외부에서 원하는 만큼 많은 종속성을 주입할 수 있으며 훨씬 더 유연해집니다. 오늘은 여기까지입니다. 다음 강의에서 이야기하겠습니다.
GitHub과 Linkedin에서 저와 소통하실 수 있습니다.
위 내용은 PHP OOP 부분 구성과 상속 및 종속성 주입 비교의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!