php反射機制用法詳解

2020-07-16 10:05:47

反射

物件導向程式設計中物件被賦予了自省的能力,而這個自省的過程就是反射。

反射,直觀理解就是根據到達地找到出發地和來源。比如,一個光禿禿的物件,我們可以僅僅通過這個物件就能知道它所屬的類、擁有哪些方法。

反射是指在PHP執行狀態中,擴充套件分析PHP程式,匯出或提出關於類、方法、屬性、引數等的詳細資訊,包括注釋。這種動態獲取資訊以及動態呼叫物件方法的功能稱為反射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 '未設定';
       
     $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]}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');

在平常開發中,用到反射的地方不多:一個是對物件進行偵錯,另一個是獲取類的資訊。在MVC和外掛開發中,使用反射很常見,但是反射的消耗也很大,在可以找到替代方案的情況下,就不要濫用。

很多時候,善用反射能保持程式碼的優雅和簡潔,但反射也會破壞類的封裝性,因為反射可以使本不應該暴露的方法或屬性被強制暴露了出來,這既是優點也是缺點。

想了解更多相關內容請存取PHP中文網:PHP視訊教學

以上就是php反射機制用法詳解的詳細內容,更多請關注TW511.COM其它相關文章!