反射
物件導向程式設計中物件被賦予了自省的能力,而這個自省的過程就是反射。
反射,直觀理解就是根據到達地找到出發地和來源。比如,一個光禿禿的物件,我們可以僅僅通過這個物件就能知道它所屬的類、擁有哪些方法。
反射是指在PHP執行狀態中,擴充套件分析PHP程式,匯出或提出關於類、方法、屬性、引數等的詳細資訊,包括注釋。這種動態獲取資訊以及動態呼叫物件方法的功能稱為反射API。
如何使用反射API?
<?php class person{ public $name; public $gender; public function say(){ echo $this->name," tis ",$this->gender,"rn"; } public function set($name, $value) { echo "Setting $name to $value rn"; $this->$name= $value; } public function get($name) { if(!isset($this->$name)){ echo '未設定'; $this->$name="正在為你設定預設值"; } return $this->$name; } } $student=new person(); $student->name='Tom'; $student->gender='male'; $student->age=24;
現在,要獲取這個student物件的方法和屬性列表該怎麼做呢?如以下程式碼所示:
// 獲取物件屬性列表 $reflect = new ReflectionObject($student); $props = $reflect->getProperties(); foreach ($props as $prop) { print $prop->getName() ."n"; } // 獲取物件方法列表 $m=$reflect->getMethods(); foreach ($m as $prop) { print $prop->getName() ."n"; }
也可以不用反射API,使用class函數,返回物件屬性的關聯陣列以及更多的資訊:
// 返回物件屬性的關聯陣列 var_dump(get_object_vars($student)); // 類屬性 var_dump(get_class_vars(get_class($student))); // 返回由類的方法名組成的陣列 var_dump(get_class_methods(get_class($student)));
假如這個物件是從其他頁面傳過來的,怎麼知道它屬於哪個類呢?一句程式碼就可以搞定:
// 獲取物件屬性列表所屬的類 echo get_class($student);
反射API的功能顯然更強大,甚至能還原這個類的原型,包括方法的存取許可權等,如:
// 反射獲取類的原型 $obj = new ReflectionClass('person'); $className = $obj->getName(); $Methods = $Properties = array(); foreach($obj->getProperties() as $v) { $Properties[$v->getName()] = $v; } foreach($obj->getMethods() as $v) { $Methods[$v->getName()] = $v; } echo "class {$className}n{n"; is_array($Properties)&&ksort($Properties); foreach($Properties as $k => $v) { echo "t"; echo $v->isPublic() ? ' public' : '',$v->isPrivate() ? ' private' : '', $v->isProtected() ? ' protected' : '', $v->isStatic() ? ' static' : ''; echo "t{$k}n"; } echo "n"; if(is_array($Methods)) ksort($Methods); foreach($Methods as $k => $v) { echo "tfunction {$k}(){}n"; } echo "}n";
輸出如下:
class person { public gender public name function get(){} function set(){} function say(){} }
不僅如此,PHP手冊中關於反射API更是有幾十個,可以說,反射完整地描述了一個類或者物件的原型。反射不僅可以用於類和物件,還可以用於函數、擴充套件模組、異常等。
反射有什麼作用?
反射可以用於文件生成。因此可以用它對檔案裡的類進行掃描,逐個生成描述文件。
既然反射可以探知類的內部結構,那麼是不是可以用它做hook實現外掛功能呢?或者是做動態代理呢?
例如:
<?php class mysql { function connect($db) { echo "連線到資料庫${db[0]}rn"; } } class sqlproxy { private $target; function construct($tar) { $this->target[] = new $tar(); } function call($name, $args) { foreach ($this->target as $obj) { $r = new ReflectionClass($obj); if ($method = $r->getMethod($name)) { if ($method->isPublic() && !$method->isAbstract()) { echo "方法前攔截記錄LOGrn"; $method->invoke($obj, $args); echo "方法後攔截rn"; } } } } } $obj = new sqlproxy('mysql'); $obj->connect('member');
在平常開發中,用到反射的地方不多:一個是對物件進行偵錯,另一個是獲取類的資訊。在MVC和外掛開發中,使用反射很常見,但是反射的消耗也很大,在可以找到替代方案的情況下,就不要濫用。
很多時候,善用反射能保持程式碼的優雅和簡潔,但反射也會破壞類的封裝性,因為反射可以使本不應該暴露的方法或屬性被強制暴露了出來,這既是優點也是缺點。
想了解更多相關內容請存取PHP中文網:PHP視訊教學
以上就是php反射機制用法詳解的詳細內容,更多請關注TW511.COM其它相關文章!