反射
面向?qū)ο缶幊讨袑?duì)象被賦予了自省的能力,而這個(gè)自省的過程就是反射。
反射,直觀理解就是根據(jù)到達(dá)地找到出發(fā)地和來源。比如,一個(gè)光禿禿的對(duì)象,我們可以僅僅通過這個(gè)對(duì)象就能知道它所屬的類、擁有哪些方法。
反射是指在php運(yùn)行狀態(tài)中,擴(kuò)展分析PHP程序,導(dǎo)出或提出關(guān)于類、方法、屬性、參數(shù)等的詳細(xì)信息,包括注釋。這種動(dòng)態(tài)獲取信息以及動(dòng)態(tài)調(diào)用對(duì)象方法的功能稱為反射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 '未設(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對(duì)象的方法和屬性列表該怎么做呢?如以下代碼所示:
// 獲取對(duì)象屬性列表
$reflect = new ReflectionObject($student);
$props = $reflect->getProperties();
foreach ($props as $prop) {
print $prop->getName() ."n";
}
// 獲取對(duì)象方法列表
$m=$reflect->getMethods();
foreach ($m as $prop) {
print $prop->getName() ."n";
}
也可以不用反射API,使用class函數(shù),返回對(duì)象屬性的關(guān)聯(lián)數(shù)組以及更多的信息:
// 返回對(duì)象屬性的關(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è)對(duì)象是從其他頁面?zhèn)鬟^來的,怎么知道它屬于哪個(gè)類呢?一句代碼就可以搞定:
// 獲取對(duì)象屬性列表所屬的類
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手冊(cè)中關(guān)于反射API更是有幾十個(gè),可以說,反射完整地描述了一個(gè)類或者對(duì)象的原型。反射不僅可以用于類和對(duì)象,還可以用于函數(shù)、擴(kuò)展模塊、異常等。
反射有什么作用?
反射可以用于文檔生成。因此可以用它對(duì)文件里的類進(jìn)行掃描,逐個(gè)生成描述文檔。
既然反射可以探知類的內(nèi)部結(jié)構(gòu),那么是不是可以用它做hook實(shí)現(xiàn)插件功能呢?或者是做動(dòng)態(tài)代理呢?
例如:
<?php
class MySQL {
function connect($db) {
echo "連接到數(shù)據(jù)庫${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');
在平常開發(fā)中,用到反射的地方不多:一個(gè)是對(duì)對(duì)象進(jìn)行調(diào)試,另一個(gè)是獲取類的信息。在MVC和插件開發(fā)中,使用反射很常見,但是反射的消耗也很大,在可以找到替代方案的情況下,就不要濫用。
很多時(shí)候,善用反射能保持代碼的優(yōu)雅和簡潔,但反射也會(huì)破壞類的封裝性,因?yàn)榉瓷淇梢允贡静粦?yīng)該暴露的方法或?qū)傩员粡?qiáng)制暴露了出來,這既是優(yōu)點(diǎn)也是缺點(diǎn)。