简单谈谈PHP中的trait
前言
之前的一个同事换工作,在面试被问到了PHP的trait。因为没用过,所以没答好,我大概是用过几次的,想了想整理了以下的总结。
trait
trait是在一些类(Class)的应该具备的特定的属性或方法,而同父级的另外一些类应该避免包含这些属性和方法情况下使用的.
当然,这也和开发者对类的抽象能力有关,有些抽象能力好的,可以减少对trait的使用但是这种情况应该是无法避免的不然trait出现就毫无意义了.
还有一种情况,就是使用trait的时候,可以起到的约束开发者的作用,提醒开发者注意需要在开发的过程中调用trait的某些属性和方法.
同事则提出了一个好问题,接口(interface)不也是这个作用么?
不急,让我们先看个例子:
比如你要收集网站上各类数据,开发了Spider类.Spider有个方法叫request()负责请求.
<?phpnamespaceXWSoul\Network; classSpider { publicfunctionrequest($url) { //dosth. } }
但是采集数据的过程中,有些网站对蜘蛛敏感有些则不.对于敏感的网站,我们给出了一个使用代理的解决方案.但是使用代理是会影响抓取速度的.这就产生了Spider的子类有些需要用代理,而能不用代理则尽量不用的情况.
于是这个时候我们新增了一个traitProxy:
<?phpnamespaceXWSoul\Network; traitProxy { protected$isProxy=false; publicfunctionuseProxy($proxy) { //dosthproxysetups. $this->isProxy=true; return$this; } publicfunctionrequest($url) { if(!$this->isProxy){ thrownewException("Pleaseusingproxy."); } //dosth. returnparent::request($url); } }
trait重写了Spider的request()方法,限定了在没有调用代理的情况下调用会抛出异常.
回到之前的问题,trait这样的用法和接口(interface)有什么区别?
接口的约束是前置的是定义初始就必须实现的,他可以约束方法的实现却无法约束方法的调用,trait是一种后置的调用,他已经实现了方法,关键的是,他只对调用了自身的类产生约束(废话一句),而对没有调用自身的类不产生影响(再一句废话),同时他是可复用的,而且没有破坏Spider类自身的实现增加,Spider还是那个Spider.
我想trait的用法再这里已经很有效了吧.
后话
有人可能决定另外实现一个request比如叫,proxyRequst不就完了么?你说的好有道理…然是如果我使用了不一样的代理具体对请求上有细节差异怎么办呢?在代码里不停的ififif么?trait如此清爽的方案为何要放弃呢?
总结
好了,以上就是这篇文章的全部内容了希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流。