PHP 反射之動態代理

pardon110發表於2019-09-22

反射可以探知類的內部結構 可以用它做hook實現外掛功能,或者做動態代理

與反射相關

  • 類和物件相關的函式
    get_object_vars
    get_class_methods
    get_class_vars
    get_class
    get_parent_class
    method_exists
    property_exists
    trait_exists
  • 反射相關的 API 類
    reflectiontype
    reflectionproperty
    reflectionobject
    reflectionfunction
    reflectionmethod
    reflectionexception
    reflectionextension
    reflectionparameter
    reflectionfunctionabstract
    reflectiongenerator
    reflectionclass
    reflectionclassconstant
    reflectionzendextension

    反射API功能更強大,甚至能還原這個類的原型,包括方法的訪問許可權等

應用場景

一個是對物件進行除錯,另一個是獲取類的資訊,通常有以下應用方式

  • 文件生成 用它對檔案裡的類進行掃描,生成描述文件
  • 外掛開發 在MVC和外掛開發中,常見使用反射

缺點

  • 反射的效能消耗也很大,一般情況下儘量不使用
  • 會破壞類的封裝性,因為反射可以使本不應該暴露的方法或屬性被強制暴露了出來

例項

下面是一個利用反射特性,實現的簡單的資料庫動態代理

基於動態代理,可以有更多的想象空間,如實現攔截器,屬性方法增加,裁剪等等

class Mysql
{
    function connect($db){
         echo "connecting database ${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 "method before record \r\n";
                     $method->invoke($obj,$args);
                     echo "method after record\r\n";
                 }
             }
         }
     }
 }
 $obj = new SqlProxy('Mysql');
 $obj->connect('member');

其它

  • echoprint 都是語言結構,但是後者有返回值
  • print_rvar_dump 是普通函式 皆可列印多種型別資料,但後者會輸出資料型別,前者第二引數可改變輸出為返回
本作品採用《CC 協議》,轉載必須註明作者和本文連結

相關文章