前言
之前的一個同事換工作,在面試被問到了 PHP 的 trait 。因為沒用過, 所以沒答好,我大概是用過幾次的,想了想整理了以下的總結。
trait
trait 是在一些類別(Class)的應該具備的特定的屬性或方法,而同父級的另外一些類別應該避免包含這些屬性和方法情況下使用的.
當然, 這也和開發者對類別的抽象能力有關, 有些抽象能力好的, 可以減少對trait 的使用但是這種情況應該是無法避免的不然trait出現就毫無意義了.
還有一種情況, 就是使用trait 的時候, 可以起到的約束開發者的作用, 提醒開發者註意需要在開發的過程中調用trait 的某些屬性和方法.
同事則提出了一個好問題, 介面(interface) 不也是這個作用麼?
不急, 讓我們先看個例子:
例如你要收集網站上各類資料, 開發了Spider 類別. Spider
有個方法叫 request()
負責請求.
<?php namespace XWSoul\Network; class Spider { public function request($url) { //do sth. } }
但是採集資料的過程中, 有些網站對蜘蛛敏感有些則不. 對於敏感的網站, 我們給出了一個使用代理的解決方案. 但是使用代理是會影響抓取速度的. 這就產生了Spider 的子類別有些需要用代理, 而能不用代理則盡量不用的情況.
於是這個時候我們新增了一個trait Proxy:
<?php namespace XWSoul\Network; trait Proxy { protected $isProxy = false; public function useProxy($proxy) { //do sth proxy setups. $this->isProxy = true; return $this; } public function request($url) { if (!$this->isProxy) { throw new Exception("Please using proxy."); } //do sth. return parent::request($url); } }
trait 重寫了Spider 的request()
方法, 限定了在沒有呼叫代理的情況下呼叫會拋出例外.
回到之前的問題, trait 這樣的用法和接口(interface) 有什麼區別?
接口的約束是前置的是定義初始就必須實現的, 他可以約束方法的實作卻無法約束方法的呼叫, trait 是一種後置的呼叫, 他已經實現了方法, 關鍵的是, 他只對調用了自身的類產生約束(廢話一句), 而對沒有調用自身的類別不產生影響(再一句廢話), 同時他是可復用的, 而且沒有破壞Spider 類別自身的實現增加, Spider 還是那個Spider.
我想trait 的用法再這裡已經很有效了吧.
後話
有人可能決定另外實作一個request 例如叫, proxyRequst 不就完了麼? 你說的好有道理…然是如果我使用了不一樣的代理具體對請求上有細節差異怎麼辦呢? 在代碼裡不停的if if if 麼? trait 如此清爽的方案為何要放棄呢?
更多PHP中的trait相關文章請關注PHP中文網!