PHP的Reflection反射機制


原文地址: http://www.nowamagic.net/php/php_Reflection.php

PHP5添加了一項新的功能:Reflection。這個功能使得程序員可以

reverse-engineer[逆向工程] class, interface,function,method and extension[擴展庫支持]。

通過PHP代碼,就可以得到某object的所有信息,並且可以和它交互。

如假設以下Person類:

 1 class Person {
 2     /**
 3      * For the sake of demonstration, we"re setting this private
 4      */
 5     private $_allowDynamicAttributes = false;
 6     
 7     /**
 8      * type=primary_autoincrement
 9      */
10     protected $id = 0;
11     
12     /**
13      * type=varchar length=255 null
14      */
15     protected $name;
16     
17     /**
18      * type=text null
19      */
20     protected $biography;
21     public function getId() {
22         return $this->id;
23     }
24     public function setId($v) {
25         $this->id = $v;
26     }
27     public function getName() {
28         return $this->name;
29     }
30     public function setName($v) {
31         $this->name = $v;
32     }
33     public function getBiography() {
34         return $this->biography;
35     }
36     public function setBiography($v) {
37         $this->biography = $v;
38     }
39 }

通過ReflectionClass,我們可以得到Person類的以下信息:

  • 常量 Contants
  • 屬性 Property Names
  • 方法 Method Names
  • 靜態屬性 Static Properties
  • 命名空間 Namespace
  • Person類是否為final或者abstract

只要把類名"Person"傳遞給ReflectionClass就可以了:

1 $class = new ReflectionClass('Person');

* 獲取屬性(Properties):

1 $properties = $class->getProperties();
2 foreach($properties as $property) {
3     echo $property->getName()."\n";
4 }
5 // 輸出:
6 // _allowDynamicAttributes
7 // id
8 // name
9 // biography

默認情況下,ReflectionClass會獲取到所有的屬性,private 和 protected的也可以。如果只想獲取到private屬性,就要額外傳個參數:

1 $private_properties = $class->getProperties(ReflectionProperty::IS_PRIVATE);

可用參數列表:

  • ReflectionProperty::IS_STATIC
  • ReflectionProperty::IS_PUBLIC
  • ReflectionProperty::IS_PROTECTED
  • ReflectionProperty::IS_PRIVATE

如果要同時獲取public 和private 屬性,就這樣寫:ReflectionProperty::IS_PUBLIC | ReflectionProperty::IS_PRIVATE

通過$property->getName()可以得到屬性名,通過getDocComment可以得到寫給property的注釋。

 1 foreach($properties as $property) {
 2     if($property->isProtected()) {
 3         $docblock = $property->getDocComment();
 4         preg_match('/ type\=([a-z_]*) /', $property->getDocComment(), $matches);
 5         echo $matches[1]."\n";
 6     }
 7 }
 8 // Output:
 9 // primary_autoincrement
10 // varchar
11 // text

有點不可思議了吧。竟然連注釋都可以取到。

* 獲取方法(methods):通過getMethods() 來獲取到類的所有methods。返回的是ReflectionMethod對象的數組。

不再演示。

 

賦一個利用反射加載類實例的工具類寫法:

<?php
/**
 * Desc: 反射加載類
 * Class: ClassLoader
 * Package: App\Lib
 * User: zb
 * Date: 2019/5/22 15:21
 */

namespace App\Lib;

class ClassLoader
{
    private static $obj = [];

    /**
     * 上傳類map
     */
    private static function uploadClassMap ()
    {
        $map = [
            'image' => 'App\Lib\Upload\Image',//key為傳遞參數key
            'video' => 'App\Lib\Upload\Video',
        ];
        return $map;
    }

    /**
     * model類map
     */
    private static function modelClassMap ()
    {
        $map = [
            'video' => 'App\Model\Video',
        ];
        return $map;
    }

    /**
     * aliyun類map
     */
    private static function aliyunClassMap ()
    {
        $map = [
            'vod' => 'App\Lib\Aliyun\AliyunVod',
        ];
        return $map;
    }

    /**
     * cache類map
     */
    private static function cacheClassMap ()
    {
        $map = [
            'video' => 'App\Lib\Cache\Video',
        ];
        return $map;
    }

    /**
     * 類實例map指南
     * @param string $guideKey 指南key
     * @param string $classKey classKey
     * @return array|bool
     */
    private static function guideClassMap (string $guideKey, string $classKey)
    {
        $guideKey = strtolower($guideKey);
        $guideMap = [
            'upload' => self::uploadClassMap(),
            'model' => self::modelClassMap(),
            'aliyun' => self::aliyunClassMap(),
            'cache' => self::cacheClassMap(),
        ];
        return isset($guideMap[$guideKey][$classKey]) ? $guideMap[$guideKey][$classKey] : array();
    }

    /**
     * @param string $guideKey guideKey
     * @param string $classKey classKey鍵
     * @param array $params 構造函數參數
     * @param bool $instance 是否需要實例化
     * @return object|string 返回obj或字符串
     * @throws \ReflectionException
     */
    public static function initClass (string $guideKey, string $classKey, $params = [], $instance = true)
    {
        if (!$params && isset(self::$obj[$guideKey . '-' . $classKey]) && is_object(self::$obj[$guideKey . '-' . $classKey])) {
            return self::$obj[$guideKey . '-' . $classKey];
        } else {
            unset(self::$obj[$guideKey . '-' . $classKey]);
        }

        if (!self::guideClassMap($guideKey, $classKey)) {
            throw new \ReflectionException('加載類不存在');
        }
        $class = self::guideClassMap($guideKey, $classKey);
        try {
            $obj = $instance ? (new \ReflectionClass($class))->newInstanceArgs($params) : $class;
            if (!$params) {
                self::$obj[$guideKey . '-' . $classKey] = $obj;
            }
            return $obj;
        } catch (\ReflectionException $e) {
            throw new \ReflectionException('類加載失敗');
        }
    }

}

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM