首页>文档>技术文档>PHP反射机制是什么?反射类型有哪些?

此组别内的文章

需要支持?

如果通过文档没办法解决您的问题,请提交工单获取我们的支持!

PHP反射机制是什么?反射类型有哪些?

这篇文章给大家分享的是PHP反射机制的内容,一些朋友可能对于PHP反射机制不是很了解,对此本文通过简要的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,都可以获取到。

  1. <?php
  2. namespace Extend;
  3.  
  4. use ReflectionClass;
  5. use Exception;
  6.  
  7. /**
  8. * 用户相关类
  9. * Class User
  10. * @package Extend
  11. */
  12. class User{
  13. const ROLE = 'Students';
  14. public $username = '';
  15. private $password = '';
  16.  
  17. public function __construct($username, $password)
  18. {
  19. $this->username = $username;
  20. $this->password = $password;
  21. }
  22.  
  23. /**
  24. * 获取用户名
  25. * @return string
  26. */
  27. public function getUsername()
  28. {
  29. return $this->username;
  30. }
  31.  
  32. /**
  33. * 设置用户名
  34. * @param string $username
  35. */
  36. public function setUsername($username)
  37. {
  38. $this->username = $username;
  39. }
  40.  
  41. /**
  42. * 获取密码
  43. * @return string
  44. */
  45. private function getPassword()
  46. {
  47. return $this->password;
  48. }
  49.  
  50. /**
  51. * 设置密码
  52. * @param string $password
  53. */
  54. private function setPassowrd($password)
  55. {
  56. $this->password = $password;
  57. }
  58. }
  59.  
  60. $class = new ReflectionClass('Extend\User'); // 将类名User作为参数,即可建立User类的反射类
  61. $properties = $class->getProperties(); // 获取User类的所有属性,返回ReflectionProperty的数组
  62. $property = $class->getProperty('password'); // 获取User类的password属性ReflectionProperty
  63. $methods = $class->getMethods(); // 获取User类的所有方法,返回ReflectionMethod数组
  64. $method = $class->getMethod('getUsername'); // 获取User类的getUsername方法的ReflectionMethod
  65. $constants = $class->getConstants(); // 获取所有常量,返回常量定义数组
  66. $constant = $class->getConstant('ROLE'); // 获取ROLE常量
  67. $namespace = $class->getNamespaceName(); // 获取类的命名空间
  68. $comment_class = $class->getDocComment(); // 获取User类的注释文档,即定义在类之前的注释
  69. $comment_method = $class->getMethod('getUsername')->getDocComment(); // 获取User类中getUsername方法的注释文档

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

    交互

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

  1. $class = new ReflectionClass('Extend\User'); // 将类名User作为参数,即可建立User类的反射类
  2. $instance = $class->newInstance('youyou', 1, '***'); // 创建User类的实例
  3.  
  4. $instance->setUsername('youyou_2'); // 调用User类的实例调用setUsername方法设置用户名
  5. $value = $instance->getUsername(); // 用过User类的实例调用getUsername方法获取用户名
  6. echo $value;echo "\n"; // 输出 youyou_2
  7.  
  8. $class->getProperty('username')->setValue($instance, 'youyou_3'); // 通过反射类ReflectionProperty设置指定实例的username属性值
  9. $value = $class->getProperty('username')->getValue($instance); // 通过反射类ReflectionProperty获取username的属性值
  10. echo $value;echo "\n"; // 输出 youyou_3
  11.  
  12. $class->getMethod('setUsername')->invoke($instance, 'youyou_4'); // 通过反射类ReflectionMethod调用指定实例的方法,并且传送参数
  13. $value = $class->getMethod('getUsername')->invoke($instance); // 通过反射类ReflectionMethod调用指定实例的方法
  14. echo $value;echo "\n"; // 输出 youyou_4
  15.  
  16. try {
  17. $property = $class->getProperty('password_1');
  18. $property->setAccessible(true); // 修改 $property 对象的可访问性
  19. $property->setValue($instance, 'password_2'); // 可以执行
  20. $value = $property->getValue($instance); // 可以执行
  21. echo $value;echo "\n"; // 输出 password_2
  22. $class->getProperty('password')->setAccessible(true); // 修改临时ReflectionProperty对象的可访问性
  23. $class->getProperty('password')->setValue($instance, 'password');// 不能执行,抛出不能访问异常
  24. $value = $class->getProperty('password')->getValue($instance); // 不能执行,抛出不能访问异常
  25. $value = $instance->password; // 不能执行,类本身的属性没有被修改,仍然是private
  26. }catch(Exception $e){echo $e;}

    注意事项

  1. 直接访问 protected 或则 private 的熟悉或者方法会抛出异常
  2. 需要调用指定的 ReflectionProperty 或则 ReflectionMethod 对象 setAccessible(true)方法才能访问非公有成员
  3. 修改非公有成员的访问权限只作用于当前的反射类的实例
  4. 需要注意获取静态成员和非静态成员所使用的方法不一样
  5. 获取父类成员的方法和一般的不一样

    关于PHP反射机制的内容就介绍到这,本文的PHP反射机制案例讲解值得参考,感兴趣的朋友可以了解看看,希望对大家有学习PHP帮助,想要了解更多大家可以关注其它的相关文章。

0 条回复 A文章作者 M管理员
欢迎您,新朋友,感谢参与互动!
    暂无讨论,说说你的看法吧
QQ客服
  • QQ176363189 点击这里给我发消息
旺旺客服
  • 速度网络服务商 点这里给我发消息
电子邮箱
  • sudu@yunjiasu.cc
微信客服
  • suduwangluo