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('类加载失败');
        }
    }

}