trait
完成一部分接口的功能,同时也完成一部分父类的功能。
举个例子来说明trait的需求场景:
当有两个class:RetailStore和Car类,这两个类,他们并不继承同一个父类,是两个不同的子类。现在有一个需求,需要这两个类都显示他们的经纬度属性。这时,两个类有了同样的功能:获取当前对象的经纬度属性。
解决这个问题有3个办法:
创建一个父类
Geocodable
,让两者都继承该类。然而这种方法并不好。因为这两个类确实是完全不相关的,并不应该继承同一祖先。创建
Geocodable
接口,让这两个类均实现这两个接口。这种方法相对好一点,两个类均能保持自己的继承层次接口,只在当前的这两个类中实现这个接口,完成相同的功能。但是,我们要在这两个类中实现相同的功能,代码有重复,这样并不好。
最好的办法:创建Geocodable
trait,定义并实现经纬度相关方法,然后把在RetailStore和Car两个类中混入这个trait。这么做,即不会破坏继承层次结构,同时又实现复用。
创建Trait
<?php
trait MyTrait{}
定义trait:
<?php
trait Geocodable
{
protected $address;
/** \Geocoder\Geocoder */
protected $geocoder;
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->getLonnitude();
}
protected function geocodeAddress()
{
$this->geocoderResult = $this->geocoder->geocode($this->address);
return true;
}
}
如何使用
<?php
class MyClass
{
use MyTrait;
}
class RetailStore
{
use Geocodable;
}
这样,每一个RetailStore类都可以使用Geocodable的特性了。
<?php
$geocoderAdapter = new \Geocoder\HttpAdapter\CurlHttpAdapter();
$geocoderProvider = new \Geocoder\Provider\GoogleMapsProvider($geocoderAdapter);
$geocoder = new\Geocoder\Geocoder($geocoderProvider);
$store = new RetailStore();
$store->setAddress('your set address');
$store->setGeocoder($geocoder);
$latitude = $store->getLatitude();
$longitude = $store->getLongitude();
echo $latitude, ':', $longitude;
参考
Modern PHP