PHP中的反射

面向對象編輯中對象被賦予了自省的能力,而這個自省的過程就是反射.php

反射,直觀理解應時根據到達地找出出發地和來源.比方說,我給你一個光禿禿的對象,我能夠僅僅經過這個對象就能知道它所屬的類,擁有哪些方法.
反射指在PHP運行狀態中,擴展分析PHP程序,導出或提取出關於類,方法,屬性,參數等詳細信息,包括註釋.這種動態獲取信息以及動態調用對象方法的功能稱爲反射API
 

如何使用反射API

如下面代碼爲例
class HandsonBoy
{
    public $name = 'chenqionghe';
    public $age = 18;
    public function __set($name,$value)
    {
        echo '您正在設置私有屬性'.$name.'<br >值爲'.$value.'<br>';
        $this->$name = $value;
    }
    public function __get($name)
    {
        if(!isset($this->$name))
        {
            echo '未設置'.$name;
            $this->$name = "正在爲你設置默認值".'<br>';
        }
        return $this->$name;
    }
}
$boy = new HandsonBoy();
echo $boy->name.'<br />';
$boy->hair = 'short';
如今,要獲取這個student對象的方法和屬性列表該怎麼作?能夠用反射來實現,代碼以下
$reflect = new ReflectionObject($boy);
$props = $reflect->getProperties();
//獲取屬性的名字
foreach($props as $prop)
{
    print $prop->getName().PHP_EOL;
}
//獲取對象方法列表
$methos = $reflect->getMethods();
foreach($methos as $method)
{
    print $method->getName().PHP_EOL;
}

也能夠不用反射API,使用class函數,返回對象屬性的關聯數組以及更多的信息:(針對於公開的屬性和):mysql

//返回對象屬性的關聯數組
var_dump(get_object_vars($boy));
//類屬性
var_dump(get_class_vars(get_class($boy)));
//返回由類的屬性的方法名組成的數組
var_dump(get_class_methods(get_class($boy)));

 反射API的功能顯然更強大,甚至能還原這個類的原型,包括方法的訪問權限,如下簡單封裝了一個打印類的代碼sql

/**
 * @param $classObject 對象或者類名
 */
function getClass($classObject)
{
    $object = new ReflectionClass($classObject);
    $className = $object->getName();
    $Methods = $Properties = array();
    foreach($object->getProperties() as $v)
    {
        $Properties[$v->getName()] = $v;
    }
    foreach($object->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";
}

 不只如此,PHP手冊中關於反射API更是有幾十個,能夠說,反射完整地描述了一個類或者對象的原型.反射不只能夠用於類和對象,還能夠用於函數,擴展模塊,異常等.數據庫

反射有什麼做用

反射能夠用於文檔生成,所以能夠用它對文件裏的類進行掃描,逐個生成描述文檔.
既然反射能夠探知類內部結構, 那麼是否是 能夠用它作hook實現插件功能呢?或者是做動態代理呢?拋磚引玉,如下代碼是個簡單的例子
<?php
class mysql
{
    function connect($db)
    {
        echo "鏈接到數據庫{$db[0]}".PHP_EOL;
    }
}
class sqlproxy
{
    private $target;
    public function __construct($tar)
    {
        $this->target[] = new $tar;
    }
    public 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".PHP_EOL;
                    $method->invoke($obj,$args);
                    echo "方法後攔截".PHP_EOL;
                }
            }
        }
    }
}
$obj = new sqlproxy('mysql');
$obj->connect('chenqionghe');
這裏真正操做類是mysql類,可是sqlproxy實現了根據動態傳入參數,代替實際的類運行,而且在方法運行先後進行攔截,而且動態地改變類中的方法和屬性.這就是簡單的動態代理.
在日常的開發中用到反射的地方並很少: 一個是對對象進行調試,別一個是獲取類的信息.在MVC和插件開發中,使用反射很常見,可是反射的消耗也很大,在能夠找到替代方案的狀況下,就不要濫用.
PHP有Token函數,能夠經過這個機制實現一些反射功能.從簡單靈活的角度講,使用已經提供的反射API是可取的.
不少時候,善用反射能保持代碼的優雅和簡潔,但反射也會破壞類的封裝性,由於反射可使本不該該暴露的方法或屬性被強制暴露了出來,這既是優勢也是缺點.
相關文章
相關標籤/搜索