Skip to main content

PHP反射机制

PHP反射机制

简介

就算是类成员定义为private也可以在外部访问,不用创建类的实例也可以访问类的成员和方法。

PHP自5.0版本以后添加了反射机制,它提供了一套强大的反射API,允许你在PHP运行环境中,访问和使用类、方法、属性、参数和注释等,其功能十分强大,经常用于高扩展的PHP框架,自动加载插件,自动生成文档,甚至可以用来扩展PHP语言。由于它是PHP內建的oop扩展,为语言本身自带的特性,所以不需要额外添加扩展或者配置就可以使用。更多内容见官方文档。

反射类型

PHP反射API会基于类,方法,属性,参数等维护相应的反射类,已提供相应的调用API。

类型说明
ReflectorReflector 是一个接口,被所有可导出的反射类所实现(implement)
Reflection反射(reflection)类
ReflectionClass报告了一个类的有关信息
ReflectionZendExtension报告Zend扩展的相关信息
ReflectionExtension报告了PHP扩展的有关信息
ReflectionFunction报告了一个函数的有关信息
ReflectionFunctionAbstractReflectionFunction 的父类
ReflectionMethod报告了一个方法的有关信息
ReflectionObject报告了一个对象(object)的相关信息
ReflectionParameter取回了函数或方法参数的相关信息
ReflectionProperty报告了类的属性的相关信息

访问

假设定义了一个类 User,我们首先需要建立这个类的反射类实例,然后基于这个实例可以访问 User 中的属性或者方法。不管类中定义的成员权限声明是否为public,都可以获取到。

<?php 
namespace Extend;

use ReflectionClass;
use Exception;

/**
* 用户相关类
* Class User
* @package Extend
*/
class User{
const ROLE = 'Students';
public $username = '';
private $password = '';

public function __construct($username, $password)
{
$this->username = $username;
$this->password = $password;
}

/**
* 获取用户名
* @return string
*/
public function getUsername()
{
return $this->username;
}

/**
* 设置用户名
* @param string $username
*/
public function setUsername($username)
{
$this->username = $username;
}

/**
* 获取密码
* @return string
*/
private function getPassword()
{
return $this->password;
}

/**
* 设置密码
* @param string $password
*/
private function setPassowrd($password)
{
$this->password = $password;
}
}

$class = new ReflectionClass('Extend\User'); // 将类名User作为参数,即可建立User类的反射类
$properties = $class->getProperties(); // 获取User类的所有属性,返回ReflectionProperty的数组
$property = $class->getProperty('password'); // 获取User类的password属性ReflectionProperty
$methods = $class->getMethods(); // 获取User类的所有方法,返回ReflectionMethod数组
$method = $class->getMethod('getUsername'); // 获取User类的getUsername方法的ReflectionMethod
$constants = $class->getConstants(); // 获取所有常量,返回常量定义数组
$constant = $class->getConstant('ROLE'); // 获取ROLE常量
$namespace = $class->getNamespaceName(); // 获取类的命名空间
$comment_class = $class->getDocComment(); // 获取User类的注释文档,即定义在类之前的注释
$comment_method = $class->getMethod('getUsername')->getDocComment(); // 获取User类中getUsername方法的注释文档

注意:创建反射类时传送的类名,必须包含完整的命名空间,即使使用了 use 关键字。否则找不到类名会抛出异常。

交互

一旦创建了反射类的实例,我们不仅可以通过反射类访问原来类的方法和属性,还能创建原来类的实例或则直接调用类里面的方法。

$class = new ReflectionClass('Extend\User');  // 将类名User作为参数,即可建立User类的反射类
$instance = $class->newInstance('youyou', 1, '***'); // 创建User类的实例

$instance->setUsername('youyou_2'); // 调用User类的实例调用setUsername方法设置用户名
$value = $instance->getUsername(); // 用过User类的实例调用getUsername方法获取用户名
echo $value;echo "\n"; // 输出 youyou_2

$class->getProperty('username')->setValue($instance, 'youyou_3'); // 通过反射类ReflectionProperty设置指定实例的username属性值
$value = $class->getProperty('username')->getValue($instance); // 通过反射类ReflectionProperty获取username的属性值
echo $value;echo "\n"; // 输出 youyou_3

$class->getMethod('setUsername')->invoke($instance, 'youyou_4'); // 通过反射类ReflectionMethod调用指定实例的方法,并且传送参数
$value = $class->getMethod('getUsername')->invoke($instance); // 通过反射类ReflectionMethod调用指定实例的方法
echo $value;echo "\n"; // 输出 youyou_4

try {
$property = $class->getProperty('password_1');
$property->setAccessible(true); // 修改 $property 对象的可访问性
$property->setValue($instance, 'password_2'); // 可以执行
$value = $property->getValue($instance); // 可以执行
echo $value;echo "\n"; // 输出 password_2
$class->getProperty('password')->setAccessible(true); // 修改临时ReflectionProperty对象的可访问性
$class->getProperty('password')->setValue($instance, 'password');// 不能执行,抛出不能访问异常
$value = $class->getProperty('password')->getValue($instance); // 不能执行,抛出不能访问异常
$value = $instance->password; // 不能执行,类本身的属性没有被修改,仍然是private
}catch(Exception $e){echo $e;}

注意事项

  1. 直接访问 protected 或则 private 的熟悉或者方法会抛出异常

  2. 需要调用指定的 ReflectionProperty 或则 ReflectionMethod 对象 setAccessible(true)方法才能访问非公有成员

  3. 修改非公有成员的访问权限只作用于当前的反射类的实例

  4. 需要注意获取静态成员和非静态成员所使用的方法不一样

  5. 获取父类成员的方法和一般的不一样

有时间会整理出反射类的API表,详细的API列表可以先查阅官方文档。

  • ReflectionClass::__construct — 初始化 ReflectionClass 类

  • ReflectionClass::export — 导出一个类

  • ReflectionClass::getConstant — 获取定义过的一个常量

  • ReflectionClass::getConstants — 获取一组常量

  • ReflectionClass::getConstructor — 获取类的构造函数

  • ReflectionClass::getDefaultProperties — 获取默认属性

  • ReflectionClass::getDocComment — 获取文档注释

  • ReflectionClass::getEndLine — 获取最后一行的行数

  • ReflectionClass::getExtension — 根据已定义的类获取所在扩展的 ReflectionExtension 对象

  • ReflectionClass::getExtensionName — 获取定义的类所在的扩展的名称

  • ReflectionClass::getFileName — 获取定义类的文件名

  • ReflectionClass::getInterfaceNames — 获取接口(interface)名称

  • ReflectionClass::getInterfaces — 获取接口

  • ReflectionClass::getMethod — 获取一个类方法的 ReflectionMethod。

  • ReflectionClass::getMethods — 获取方法的数组

  • ReflectionClass::getModifiers — 获取类的修饰符

  • ReflectionClass::getName — 获取类名

  • ReflectionClass::getNamespaceName — 获取命名空间的名称

  • ReflectionClass::getParentClass — 获取父类

  • ReflectionClass::getProperties — 获取一组属性

  • ReflectionClass::getProperty — 获取类的一个属性的 ReflectionProperty

  • ReflectionClass::getReflectionConstant — Gets a ReflectionClassConstant for a class's constant

  • ReflectionClass::getReflectionConstants — Gets class constants

  • ReflectionClass::getShortName — 获取短名

  • ReflectionClass::getStartLine — 获取起始行号

  • ReflectionClass::getStaticProperties — 获取静态(static)属性

  • ReflectionClass::getStaticPropertyValue — 获取静态(static)属性的值

  • ReflectionClass::getTraitAliases — 返回 trait 别名的一个数组

  • ReflectionClass::getTraitNames — 返回这个类所使用 traits 的名称的数组

  • ReflectionClass::getTraits — 返回这个类所使用的 traits 数组

  • ReflectionClass::hasConstant — 检查常量是否已经定义

  • ReflectionClass::hasMethod — 检查方法是否已定义

  • ReflectionClass::hasProperty — 检查属性是否已定义

  • ReflectionClass::implementsInterface — 接口的实现

  • ReflectionClass::inNamespace — 检查是否位于命名空间中

  • ReflectionClass::isAbstract — 检查类是否是抽象类(abstract)

  • ReflectionClass::isAnonymous — 检查类是否是匿名类

  • ReflectionClass::isCloneable — 返回了一个类是否可复制

  • ReflectionClass::isFinal — 检查类是否声明为 final

  • ReflectionClass::isInstance — 检查类的实例

  • ReflectionClass::isInstantiable — 检查类是否可实例化

  • ReflectionClass::isInterface — 检查类是否是一个接口(interface)

  • ReflectionClass::isInternal — 检查类是否由扩展或核心在内部定义

  • ReflectionClass::isIterable — Check whether this class is iterable

  • ReflectionClass::isIterateable — 检查是否可迭代(iterateable)

  • ReflectionClass::isSubclassOf — 检查是否为一个子类

  • ReflectionClass::isTrait — 返回了是否为一个 trait

  • ReflectionClass::isUserDefined — 检查是否由用户定义的

  • ReflectionClass::newInstance — 从指定的参数创建一个新的类实例

  • ReflectionClass::newInstanceArgs — 从给出的参数创建一个新的类实例。

  • ReflectionClass::newInstanceWithoutConstructor — 创建一个新的类实例而不调用它的构造函数

  • ReflectionClass::setStaticPropertyValue — 设置静态属性的值

  • ReflectionClass::__toString — 返回 ReflectionClass 对象字符串的表示形式。