PHP反射機(jī)制原理與用法詳解
本文實(shí)例講述了PHP反射機(jī)制原理與用法。分享給大家供大家參考,具體如下:
反射
面向?qū)ο缶幊讨袑ο蟊毁x予了自省的能力,而這個(gè)自省的過程就是反射。
反射,直觀理解就是根據(jù)到達(dá)地找到出發(fā)地和來源。比如,一個(gè)光禿禿的對象,我們可以僅僅通過這個(gè)對象就能知道它所屬的類、擁有哪些方法。
反射是指在PHP運(yùn)行狀態(tài)中,擴(kuò)展分析PHP程序,導(dǎo)出或提出關(guān)于類、方法、屬性、參數(shù)等的詳細(xì)信息,包括注釋。這種動態(tài)獲取信息以及動態(tài)調(diào)用對象方法的功能稱為反射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 '未設(shè)置'; $this->$name="正在為你設(shè)置默認(rèn)值"; } return $this->$name; } } $student=new person(); $student->name='Tom'; $student->gender='male'; $student->age=24;
現(xiàn)在,要獲取這個(gè)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函數(shù),返回對象屬性的關(guān)聯(lián)數(shù)組以及更多的信息:
// 返回對象屬性的關(guān)聯(lián)數(shù)組 var_dump(get_object_vars($student)); // 類屬性 var_dump(get_class_vars(get_class($student))); // 返回由類的方法名組成的數(shù)組 var_dump(get_class_methods(get_class($student)));
假如這個(gè)對象是從其他頁面?zhèn)鬟^來的,怎么知道它屬于哪個(gè)類呢?一句代碼就可以搞定:
// 獲取對象屬性列表所屬的類 echo get_class($student);
反射API的功能顯然更強(qiáng)大,甚至能還原這個(gè)類的原型,包括方法的訪問權(quán)限等,如:
// 反射獲取類的原型 $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手冊中關(guān)于反射API更是有幾十個(gè),可以說,反射完整地描述了一個(gè)類或者對象的原型。反射不僅可以用于類和對象,還可以用于函數(shù)、擴(kuò)展模塊、異常等。
反射有什么作用
反射可以用于文檔生成。因此可以用它對文件里的類進(jìn)行掃描,逐個(gè)生成描述文檔。
既然反射可以探知類的內(nèi)部結(jié)構(gòu),那么是不是可以用它做hook實(shí)現(xiàn)插件功能呢?或者是做動態(tài)代理呢?
例如:
<?php class mysql { function connect($db) { echo "連接到數(shù)據(jù)庫${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');
在平常開發(fā)中,用到反射的地方不多:一個(gè)是對對象進(jìn)行調(diào)試,另一個(gè)是獲取類的信息。在MVC和插件開發(fā)中,使用反射很常見,但是反射的消耗也很大,在可以找到替代方案的情況下,就不要濫用。
PHP有Token函數(shù),可以通過這個(gè)機(jī)制實(shí)現(xiàn)一些反射功能。從簡單靈活的角度講,使用已經(jīng)提供的反射API是可取的。
很多時(shí)候,善用反射能保持代碼的優(yōu)雅和簡潔,但反射也會破壞類的封裝性,因?yàn)榉瓷淇梢允贡静粦?yīng)該暴露的方法或?qū)傩员粡?qiáng)制暴露了出來,這既是優(yōu)點(diǎn)也是缺點(diǎn)。
更多關(guān)于PHP相關(guān)內(nèi)容感興趣的讀者可查看本站專題:《php面向?qū)ο蟪绦蛟O(shè)計(jì)入門教程》、《PHP基本語法入門教程》、《PHP運(yùn)算與運(yùn)算符用法總結(jié)》、《PHP網(wǎng)絡(luò)編程技巧總結(jié)》、《PHP數(shù)組(Array)操作技巧大全》、《php字符串(string)用法總結(jié)》、《php+mysql數(shù)據(jù)庫操作入門教程》及《php常見數(shù)據(jù)庫操作技巧匯總》
希望本文所述對大家PHP程序設(shè)計(jì)有所幫助。
相關(guān)文章
php多層數(shù)組與對象的轉(zhuǎn)換實(shí)例代碼
通過json_decode(json_encode($object)可以將對象一次性轉(zhuǎn)換為數(shù)組,但是object中遇到非utf-8編碼的非ascii字符則會出現(xiàn)問題,比如gbk的中文,何況json_encode和decode的性能也值得疑慮2013-08-08提高PHP性能的編碼技巧以及性能優(yōu)化詳細(xì)解析
include文件時(shí)盡量使用絕對路徑,因?yàn)樗苊饬薖HP去include_path里查找文件的速度,解析操作系統(tǒng)路徑所需的時(shí)間會更少2013-08-08PHP使用strrev翻轉(zhuǎn)中文亂碼問題的解決方法
這篇文章主要介紹了PHP使用strrev翻轉(zhuǎn)中文亂碼問題的解決方法,通過自定義函數(shù)遍歷字符串并設(shè)置編碼格式解決亂碼問題,需要的朋友可以參考下2017-01-01PHP.ini中配置屏蔽錯(cuò)誤信息顯示和保存錯(cuò)誤日志的例子
這篇文章主要介紹了PHP.ini中配置屏蔽錯(cuò)誤信息顯示和保存錯(cuò)誤日志的例子,需要的朋友可以參考下2014-05-05關(guān)于Anemometer圖形化顯示MySQL慢日志的工具搭建及使用的詳細(xì)介紹
這篇文章主要介紹了關(guān)于Anemometer圖形化顯示MySQL慢日志的工具搭建及使用的詳細(xì)介紹,Anemometer 是一個(gè)圖形化顯示MySQL慢日志的工具,Anemometer可以很輕松的幫你去分析慢查詢?nèi)罩?,讓你很容易就能找到哪些SQL需要優(yōu)化。感興趣可以來了解一下2020-07-07PHP 實(shí)現(xiàn) WebSocket 協(xié)議原理與應(yīng)用詳解
這篇文章主要介紹了PHP 實(shí)現(xiàn) WebSocket 協(xié)議,結(jié)合具體實(shí)例形式較為詳細(xì)的分析了websocket協(xié)議原理、以及PHP具體應(yīng)用相關(guān)操作技巧,需要的朋友可以參考下2020-04-04