上一篇《手写PHP API框架之Composer的安装使用(二)》文章中我们介绍了Composer的安装使用,这一文我们来介绍一下有关反射的概念介绍。
反射,直观理解就是根据到达地找到出发地和来源。 反射指在PHP运行状态中,扩展分析PHP程序,导出或提出关于类、方法、属性、参数等的详细信息,包括注释。这种动态获取信息以及动态调用对象方法的功能称为反射API。
不妨先来看一个demo:
<?php function p($msg, $var) { echo($msg.":".print_r($var, true)).PHP_EOL.PHP_EOL; } class Demo { private $id; protected $name; public $skills = []; public function __construct($id, $name, $skills = []) { $this->id = $id; $this->name = $name; $this->skills = $skills; } public function getName() { return $this->name; } public function getSkill() { p('skill', $this->skills); } } $ref = new ReflectionClass('Demo'); if ($ref->isInstantiable()) { p('检查类是否可实例化isInstantiable', null); } $constructor = $ref->getConstructor(); p('获取构造函数getConstructor', $constructor); $parameters = $constructor->getParameters(); foreach ($parameters as $param) { p('获取参数getParameters', $param); } if ($ref->hasProperty('name')) { $attr = $ref->getProperty('name'); p('获取属性getProperty', $attr); } $attributes = $ref->getProperties(); foreach ($attributes as $row) { p('获取属性列表getProperties', $row->getName()); } if ($ref->hasMethod('getSkill')) { $method = $ref->getMethod('getSkill'); p('获取方法getMethod', $method); } $methods = $ref->getMethods(); foreach ($methods as $row) { p('获取方法列表getMethods', $row->getName()); } $instance = $ref->newInstanceArgs([1, 'sai', ['php', 'js']]); p('newInstanceArgs', $instance);
输出:
➜ php git:(main) php reflect.php 检查类是否可实例化isInstantiable: 获取构造函数getConstructor:ReflectionMethod Object ( [name] => __construct [class] => Demo ) 获取参数getParameters:ReflectionParameter Object ( [name] => id ) 获取参数getParameters:ReflectionParameter Object ( [name] => name ) 获取参数getParameters:ReflectionParameter Object ( [name] => skills ) 获取属性getProperty:ReflectionProperty Object ( [name] => name [class] => Demo ) 获取属性列表getProperties:id 获取属性列表getProperties:name 获取属性列表getProperties:skills 获取方法getMethod:ReflectionMethod Object ( [name] => getSkill [class] => Demo ) 获取方法列表getMethods:__construct 获取方法列表getMethods:getName 获取方法列表getMethods:getSkill newInstanceArgs:Demo Object ( [id:Demo:private] => 1 [name:protected] => sai [skills] => Array ( [0] => php [1] => js ) )
demo里面就有使用了ReflectionClass类,当然ReflectionClass类不止于这些方法。
更多方法
ReflectionClass类还有更多方法:
方法 | 说明 |
---|---|
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 对象字符串的表示形式。 |
除去强大的ReflectionClass,还有Reflection、ReflectionClassConstant 、ReflectionMethod 、ReflectionFunctionAbstract等等。建议查看手册:
反射的实际应用
反射可以用于文档、文件生成。可以用它对文件里的类进行扫描,逐个生成描述文档;
既然反射可以探知类的内部结构,那么可以用它做hook实现插件功能;
可以用于做动态代理,在未知或者不确定类名的情况下,动态生成和实例化一些类和执行方法;
依赖注入。对于多次继承的类,我们可以通过多次反射探索到基类的结构,或者采用递归的形式反射,实现实例化所有继承类,这也是PHP依赖注入的原理。
反射的优点
支持反射的语言提供了一些在低级语言中难以实现的运行时特性。
可以在一定程度上避免硬编码,提供灵活性和通用性。
可以作为一个第一类对象发现并修改源代码的结构(如代码块、类、方法、协议等)。
可以在运行时像对待源代码语句一样计算符号语法的字符串(类似JavaScript的eval()函数),进而可将跟class或function匹配的字符串转换成class或function的调用或引用。
可以创建一个新的语言字节码解释器来给编程结构一个新的意义或用途。
反射的缺点
学习成本高。面向反射的编程需要较多的高级知识,包括框架、关系映射和对象交互,以利用更通用的代码执行
同样因为反射的概念和语法都比较抽象,过多地滥用反射技术会使得代码难以被其他人读懂,不利于合作与交流
反射在提高了代码灵活性的同时,牺牲了一点点运行效率,有一定的消耗
反射也会破坏类的封装性,把本不该暴露的方法或属性暴露了出来
在平时的开发中,我们用到反射其实不多,为什么把它拿到这里来说呢?一来是我们后面会使用到反射去实现Ioc容器,二来反射也是PHP核心功能之一,在我们流行的框架中十分常见,理解它是很有必要的。
这一节是比较独立的,在后面的章节中我们会使用它。
推荐学习:《PHP视频教程》
以上是手写PHP API框架(三)之反射介绍的详细内容。更多信息请关注PHP中文网其他相关文章!

PHP在现代编程中仍然是一个强大且广泛使用的工具,尤其在web开发领域。1)PHP易用且与数据库集成无缝,是许多开发者的首选。2)它支持动态内容生成和面向对象编程,适合快速创建和维护网站。3)PHP的性能可以通过缓存和优化数据库查询来提升,其广泛的社区和丰富生态系统使其在当今技术栈中仍具重要地位。

在PHP中,弱引用是通过WeakReference类实现的,不会阻止垃圾回收器回收对象。弱引用适用于缓存系统和事件监听器等场景,需注意其不能保证对象存活,且垃圾回收可能延迟。

\_\_invoke方法允许对象像函数一样被调用。1.定义\_\_invoke方法使对象可被调用。2.使用$obj(...)语法时,PHP会执行\_\_invoke方法。3.适用于日志记录和计算器等场景,提高代码灵活性和可读性。

Fibers在PHP8.1中引入,提升了并发处理能力。1)Fibers是一种轻量级的并发模型,类似于协程。2)它们允许开发者手动控制任务的执行流,适合处理I/O密集型任务。3)使用Fibers可以编写更高效、响应性更强的代码。

PHP社区提供了丰富的资源和支持,帮助开发者成长。1)资源包括官方文档、教程、博客和开源项目如Laravel和Symfony。2)支持可以通过StackOverflow、Reddit和Slack频道获得。3)开发动态可以通过关注RFC了解。4)融入社区可以通过积极参与、贡献代码和学习分享来实现。

PHP和Python各有优势,选择应基于项目需求。1.PHP适合web开发,语法简单,执行效率高。2.Python适用于数据科学和机器学习,语法简洁,库丰富。

PHP不是在消亡,而是在不断适应和进化。1)PHP从1994年起经历多次版本迭代,适应新技术趋势。2)目前广泛应用于电子商务、内容管理系统等领域。3)PHP8引入JIT编译器等功能,提升性能和现代化。4)使用OPcache和遵循PSR-12标准可优化性能和代码质量。

PHP的未来将通过适应新技术趋势和引入创新特性来实现:1)适应云计算、容器化和微服务架构,支持Docker和Kubernetes;2)引入JIT编译器和枚举类型,提升性能和数据处理效率;3)持续优化性能和推广最佳实践。


热AI工具

Undresser.AI Undress
人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover
用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

AI Hentai Generator
免费生成ai无尽的。

热门文章

热工具

禅工作室 13.0.1
功能强大的PHP集成开发环境

WebStorm Mac版
好用的JavaScript开发工具

螳螂BT
Mantis是一个易于部署的基于Web的缺陷跟踪工具,用于帮助产品缺陷跟踪。它需要PHP、MySQL和一个Web服务器。请查看我们的演示和托管服务。

SublimeText3 Linux新版
SublimeText3 Linux最新版

记事本++7.3.1
好用且免费的代码编辑器