특성
많은 PHP 개발자 친구들은 PHP 5.4.0에 도입된 새로운 개념인 특성에 대해 잘 모릅니다. 특성은 인터페이스처럼 보이지만 클래스처럼 작동합니다. 그렇다면 특성은 정확히 무엇입니까? 둘 다 아니다.
특성에는 하나 이상의 실제 PHP 클래스에 포함될 수 있는 부분 구현(예: 상수, 속성 및 메서드)이 있습니다. 특성에는 두 가지 책임이 있습니다. 클래스가 수행할 수 있는 작업을 표시하고(인터페이스와 유사) 모듈식 구현을 제공합니다(클래스와 유사).
당신은 이미 다른 언어의 특성을 어느 정도 이해하고 있을 수도 있습니다. 예를 들어 Ruby의 모듈과 믹스인의 기능은 PHP의 특성과 매우 유사합니다.
특성을 사용해야 하는 이유
PHP 언어는 고전적인 상속 모델을 사용합니다. 이는 기본 구현을 제공하는 공통 루트 클래스로 시작한다는 의미입니다. 루트 클래스에서는 상위 클래스의 다양한 구현을 직접 상속하는 보다 구체적인 클래스가 생성됩니다. 이를 상속 계층 구조라고 하며 많은 프로그래밍 언어가 이 공통 패턴을 사용합니다.
이해를 쉽게 하기 위해 생물학을 공부하기 위해 고등학교 시절로 시간여행을 떠났다고 가정해 보겠습니다. 당신이 연구한 유기체의 문, 목, 과, 속, 종을 아직도 기억합니까? 총 6개의 주요 영역이 있습니다. 영역은 문에서 파생되고, 문은 클래스에서 파생되고, 클래스는 목에서 파생되고, 목은 과에서 파생되고, 과는 속에서 파생됩니다. 속(genus) 뒤에 종(species)이 옵니다. 종 계층 구조의 각 하강은 특정 특성을 나타냅니다.
클래식 상속 모델은 대부분의 경우에 잘 작동합니다. 그러나 유사한 동작을 구현해야 하는 관련되지 않은 두 개의 클래스가 있는 경우에는 어떻게 될까요? 예를 들어, 한 PHP 클래스는 RetailStore라고 하고 다른 PHP 클래스는 Car라고 합니다. 이 두 클래스는 전혀 관련이 없는 두 클래스라고 할 수 있으며 상속 관계 측면에서 공통 상위 클래스를 공유할 수 없습니다. 그러나 두 클래스 모두 지도 좌표를 표시하려면 지리적 위치의 경도와 위도가 필요합니다.
우리는 이 문제를 해결하기 위해 특성을 만들었습니다. 관련되지 않은 클래스에 부분 구현을 주입할 수 있습니다. 또한 특성은 코드 재사용을 용이하게 합니다.
이 문제가 발생했을 때 내 첫 번째 해결책(그리고 최악의 해결책)은 RetailStore 및 Car에서 상속할 공개 상위 클래스 Geocodable을 만드는 것이었습니다. 이 해결책은 정말 좋지 않습니다. 왜냐하면 관련되지 않은 두 클래스가 공통 조상을 공유하도록 강제하는 것은 각각의 상속 계층에서 매우 어색해 보이기 때문입니다.
두 번째 해결책(약간 더 나은)은 위치정보를 구현하는 데 필요한 방법을 정의하는 Geocodable 인터페이스를 만드는 것입니다. RetialStore 및 Car 클래스 모두 이 Geocodable 인터페이스를 구현할 수 있습니다. 각 클래스가 자연스러운 상속 관계를 유지하도록 허용하는 것은 실제로 좋은 솔루션입니다. 하지만 여전히 각 클래스의 인터페이스에서 정의를 반복해야 하는데 이는 DRY 솔루션이 아닙니다.
DRY는 Don't Repeat yourself의 약자입니다. 좋은 프로그래밍 습관으로서, 여러 곳에서 동일한 코드를 반복해서는 안 됩니다. 코드 한 부분을 변경했기 때문에 다른 곳에서 동일한 코드를 수동적으로 수정해야 하는 상황은 있을 수 없습니다.
세 번째 솔루션(최고의 솔루션)은 Geocodable 특성을 구성하고 그 안에 관련 메서드를 정의하고 구현하는 것입니다. 클래스 상속 계층 구조를 방해하지 않고 RetailStore 클래스와 Car 클래스에 Geocodable 특성을 추가할 수 있습니다.
특성 구성 방법
다음은 PHP 특성을 정의하는 방법을 보여줍니다.
<?php trait MyTrait { // 此处是trait的具体实现 }
특성의 사용을 더 잘 보여주기 위해 Geocodable 예제로 돌아가겠습니다. RetailStore 클래스와 Car 클래스가 지리적 위치 지정 기능을 지원해야 한다는 점은 모두 알고 있으며 상속과 인터페이스가 최선의 솔루션이 아니라는 점에는 모두 동의할 수 있습니다. 대신 지도에 표시할 수 있는 경도 및 위도 좌표를 반환하는 Geocodable 특성을 생성합니다. 예제 2-12는 완전한 Geocodable 특성을 보여줍니다.
예 2-12 지오코딩 특성 정의
<span style="font-size:14px;"><?php trait Geocodable { /** @var string */ protected $address; /** @var \Geocoder\Geocoder */ protected $geocoder; /** @var \Geocoder\Result\Geocoded */ protected $geocoderResult; public function setGeocoder(\Geocoder\GeocoderInterface $geocoder) { $this->geocoder = $geocoder; } public function setAddress($address) { $this->address = $address; } public function getLatitude() { if (isset($this->geocoderResult) === false) { $this->geocodeAddress(); } return $this->geocoderResult->getLatitude(); } public function getLongitude() { if (isset($this->geocoderResult) === false) { $this->geocodeAddress(); } return $this->geocoderResult->getLongitude(); } protected function geocodeAddress() { $this->geocoderResult = $this->geocoder->geocode($this->address); return true; } }</span>
지오코딩 가능 특성은 세 가지 클래스의 속성을 정의합니다.
계속...
위의 내용은 [Modern PHP] Chapter 2의 새로운 기능 세 가지 특징을 소개하며 PHP 튜토리얼에 관심이 있는 친구들에게 도움이 되기를 바랍니다.