反射
物件導向程式設計中物件被賦予了自省的能力,而這個自省的過程就是反射。
反射,直覺理解就是根據到達地找到出發地和來源。例如,一個光禿禿的對象,我們可以光是這個對象就能知道它所屬的類別、擁有哪些方法。
反射是指在PHP運行狀態中,擴展分析PHP程序,導出或提出關於類別、方法、屬性、參數等的詳細信息,包括註釋。這種動態擷取資訊以及動態呼叫物件方法的功能稱為反射API。
如何使用反射API?
<?php class person{ public $name; public $gender; public function say(){ echo $this->name," \tis ",$this->gender,"\r\n"; } public function set($name, $value) { echo "Setting $name to $value \r\n"; $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]}\r\n"; } } 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 "方法前拦截记录LOG\r\n"; $method->invoke($obj, $args); echo "方法后拦截\r\n"; } } } } } $obj = new sqlproxy('mysql'); $obj->connect('member');
在平常開發中,用到反射的地方不多:一個是對物件進行調試,另一個是獲取類別的資訊。在MVC和插件開發中,使用反射很常見,但是反射的消耗量也很大,在可以找到替代方案的情況下,就不要濫用。
很多時候,善用反射能保持程式碼的優雅和簡潔,但反射也會破壞類別的封裝性,因為反射可以使本不應該暴露的方法或屬性被強制暴露了出來,這既是優點也是缺點。
想了解更多相關內容請造訪PHP中文網:PHP影片教學
#以上是php反射機制用法詳解的詳細內容。更多資訊請關注PHP中文網其他相關文章!