PHP反射原理的实现
反射
反射,直观理解就是根据到达地找到出发地和来源。我们可以仅仅通过一个光秃秃对象就能知道它所属的类、拥有哪些方法。
反射是指在PHP运行状态中,扩展分析PHP程序,导出或提出关于类、方法、属性、参数等的详细信息,包括注释。这种动态获取信息以及动态调用对象方法的功能称为反射API。
反射其实不难理解,我们先举个反射示例
<?php
class person{
public $name;
public $gender;
public function say(){
echo $this->name," \tis ",$this->gender,"\r\n";
}
public function set($name, $value) {
echo "Setting $name to $value \r\n";
$this->$name= $value;
}
public function get($name) {
if(!isset($this->$name)){
echo '未设置默认值';
$this->$name="正在为你设置默认值2018新年快乐";
}
return $this->$name;
}
}
$student=new person();
$student->name='Tom';
$student->gender='male';
$student->age=24;
以上示例只是简单赋值,实例化,现在,要获取这个student对象的方法和属性列表该怎么做呢?如以下代码所示:
//反射实现
// 获取对象属性列表
$reflect = new ReflectionObject($student);
$props= $reflect->getProperties();
echo'获取对象属性列表'."\n";
foreach ($props as $prop) {
print $prop->getName() ."\n";
}
// 获取对象方法列表
echo'获取对象方法列表'."\n";
$m=$reflect->getMethods();
foreach ($m as $prop) {
print $prop->getName() ."\n";
}
输出:
获取对象属性列表
name
gender
age
获取对象方法列表
say
set
get
也可以不用反射API,使用class函数,返回对象属性的关联数组以及更多的信息:
// 返回对象属性的关联数组
print_r(get_object_vars($student)); // 类属性
print_r(get_class_vars(get_class($student)));
// 返回由类的方法名组成的数组
输出:
Array
(
[name] => Tom
[gender] => male
[age] =>
)
Array
(
[name] =>
[gender] =>
)
假如这个对象是从其他页面传过来的,怎么知道它属于哪个类呢?一句代码就可以搞定:
// 获取对象属性列表所属的类
echo get_class($student);//person
反射API的功能显然更强大,甚至能还原这个类的原型,包括方法的访问权限等,如:
// 反射获取类的原型
$obj = new ReflectionClass('person');
$className = $obj->getName();
$Methods = $Properties = array();
foreach($obj->getProperties() as $v)
{
$Propertienes[$v->getName()] = $v;
}
foreach($obj->getMethods() as $v) {
$Methods[$v->getName()] = $v;
}
echo "class {$className}\n{\n";
is_array($Properties)&&ksort($Properties);
foreach($Properties as $k => $v)
{
echo "\t";
echo $v->isPublic() ? ' public' : '',$v->isPrivate() ? 'private' : '',
$v->isProtected() ? 'protected':'',
$v->isStatic() ? 'static' : '';
echo "\t{$k}\n";
}
echo "\n";
if(is_array($Methods)) ksort($Methods);
foreach($Methods as $k => $v)
{
echo "\tfunction {$k}(){}\n";
}
echo "}\n";
输出:
class person
{
function get(){}
function say(){}
function set(){}
}
不仅如此,PHP手册中关于反射API更是有几十个,可以说,反射完整地描述了一个类或者对象的原型。反射不仅可以用于类和对象,还可以用于函数、扩展模块、异常等。
默认情况下,ReflectionClass会获取到所有的属性,private 和 protected的也可以。如果只想获取到private属性,就要额外传个参数:
$private_properties = $r->getProperties(ReflectionProperty::IS_PRIVATE);
可用参数列表:
- ReflectionProperty::IS_STATIC
- ReflectionProperty::IS_PUBLIC
- ReflectionProperty::IS_PROTECTED
- ReflectionProperty::IS_PRIVATE
如果要同时获取public 和private 属性,就这样写:
ReflectionProperty::IS_PUBLIC | ReflectionProperty::IS_PROTECTED
通过$property->getName()可以得到属性名,通过getDocComment可以得到写给property的注释。我没改下person完整如下:
<?php
class person{ /**
* type=primary_autoincrement 注释哦
*/
protected $id = 0;
/**
* type=varchar length=25 null
*/
public $name;
/**
* type=text null
*/
public $gender;
public function say(){
echo $this->name," \tis ",$this->gender,"\r\n";
}
public function set($name, $value) {
echo "Setting $name to $value \r\n";
$this->$name= $value;
}
public function get($name) {
if(!isset($this->$name)){
echo '未设置默认值';
$this->$name="正在为你设置默认值2018新年快乐";
}
return $this->$name;
}
}
$student=new person();
$student->name='Tom';
$student->gender='male';
$student->age=24; $class = new ReflectionClass('Person');
$properties = $class->getProperties();
foreach($properties as $property) {
if($property->isProtected()) {
$docblock = $property->getDocComment();
echo $property->getDocComment()."\n";
}
}
// 输出:
// _allowDynamicAttributes
// id
// name
// biography
输出:
/**
* type=primary_autoincrement 注释哦
*/
有点不可思议了吧。竟然连注释都可以取到。
* 获取方法(methods):通过getMethods() 来获取到类的所有methods。返回的是ReflectionMethod对象的数组。
不再演示。
反射有什么作用
反射可以用于文档生成。因此可以用它对文件里的类进行扫描,逐个生成描述文档。
既然反射可以探知类的内部结构,那么是不是可以用它做hook实现插件功能呢?或者是做动态代理呢?
<?php
class mysql {
function connect($db) {
echo "连接到数据库${db[0]}\r\n";
}
}
class sqlproxy {
private $target;
function construct($tar) {
$this->target[]= new $tar();
}
function call($name, $args) {
foreach ($this->target as $obj) {
$r = new ReflectionClass($obj);
if ($method = $r->getMethod($name)){
if ($method->isPublic() && !$method->isAbstract()) {
echo "方法前拦截记录LOG\r\n";
$method->invoke($obj, $args);
echo "方法后拦截\r\n";
}
}
}
}
}
$obj = new sqlproxy('mysql');
$obj->connect('member');
在平常开发中,用到反射的地方不多:一个是对对象进行调试,另一个是获取类的信息。在MVC和插件开发中,使用反射很常见,但是反射的消耗也很大,在可以找到替代方案的情况下,就不要滥用。
PHP有Token函数,可以通过这个机制实现一些反射功能。从简单灵活的角度讲,使用已经提供的反射API是可取的。
很多时候,善用反射能保持代码的优雅和简洁,但反射也会破坏类的封装性,因为反射可以使本不应该暴露的方法或属性被强制暴露了出来,这既是优点也是缺点。
PHP反射原理的实现的更多相关文章
- python反射原理
1.反射原理 通过字符串的形式导入模块: __import__(),可以以字符串的形式导入模块. 通过字符串的形式导入函数: 反射: 根据字符串去某个对象里面取东西,可以是字符串,函数,数字. 根据字 ...
- 简单模拟struts2核心控制器利用反射原理实现参数传递和物理视图跳转
在能够运用struts2框架进行简单网站开发后,对struts2的一些较原框架强大的功能希望有更深刻的理解,于是尝试从底层开始摸索,本文就在重新学习struts2后,利用简单代码对核心控制器的主要功能 ...
- JAVA反射原理
什么是反射? 反射,一种计算机处理方式.是程序可以访问.检测和修改它本身状态或行为的一种能力.java反射使得我们可以在程序运行时动态加载一个类,动态获取类的基本信息和定义的方法,构造函数,域等.除了 ...
- go反射原理
go反射原理 本文基于go1.13.15 1.go汇编 1.1 基本语法 go采用plan9的汇编器完成汇编,有下面几个重要的伪寄存器 FP: Frame pointer: 局部变量访问 PC: Pr ...
- [转]Java Spring的Ioc控制反转Java反射原理
转自:http://www.kokojia.com/article/12598.html 学习一个东西的时候,如果想弄明白,最好想想框架内部是如何实现的,如果是我做我会怎么实现.下面我就写一个Ioc ...
- 利用Java针对MySql封装的jdbc框架类 JdbcUtils 完整实现(包含增删改查、JavaBean反射原理,附源码)
最近看老罗的视频,跟着完成了利用Java操作MySql数据库的一个框架类JdbcUtils.java,完成对数据库的增删改查.其中查询这块,包括普通的查询和利用反射完成的查询,主要包括以下几个函数接口 ...
- 在一般处理程序中,把Form Post过来的表单集合转换成对象 ,仿 MVC post,反射原理
using System; using System.Collections.Generic; using System.Collections.Specialized; using System.L ...
- c#用反射原理递归遍历复杂实体对象
之前在网上看到的都是遍历那种比较简单的实体对象,但是如果有实体嵌套,甚至是包含有List<XXInfo>这种属性的时候就没有办法处理了.通过递归遍历的方式可以完成对复杂实体对象的所有属性的 ...
- C#中的反射原理及应用(转)
反射的概述 反射的定义:审查元数据并收集关于它的类型信息的能力.元数据(编译以后的最基本数据单元)就是一大堆的表,当编译程序集或者模块时,编译器会创建一个类定义表,一个字段定义表,和一个方法定义表等, ...
- 利用Java针对MySql封装的jdbc框架类 JdbcUtils 完整实现(包括增删改查、JavaBean反射原理,附源代码)
近期看老罗的视频,跟着完毕了利用Java操作MySql数据库的一个框架类JdbcUtils.java,完毕对数据库的增删改查.当中查询这块,包含普通的查询和利用反射完毕的查询,主要包含以下几个函数接口 ...
随机推荐
- BUAA-OO-第一单元总结
BUAA-OO第一单元博客总结 第一次作业总结 (1)类关系图 第一次作业类图关系简单,仅有一个Poly封装类以及一个Main主类调用Poly,Poly封装类内部完成了包括对象构造,求导,生成字符串的 ...
- Git Bash for Windows add ssh key时报Could not open a connection to your authentication agent.
$ ssh-add id_rsa_bitbucketCould not open a connection to your authentication agent. 运行: $ ssh-agent ...
- 一个自己研究出来的字符串匹配算法-k子串算法
前言 最近工作中需要写一个算法,而写完这个算法我却发现了一个很有意思的事情.需要的这个算法是这样的:对于A,B两个字符串,找出最多K个公共子串,使得这K个子串长度和最大.百度之没有这样的算法,然后就开 ...
- CAS实现单点登录SSO执行原理及部署
一.不落俗套的开始 1.背景介绍 单点登录:Single Sign On,简称SSO,SSO使得在多个应用系统中,用户只需要登录一次就可以访问所有相互信任的应用系统. CAS框架:CAS(Centra ...
- WordPress外链新窗口打开并使用php页面go跳转
之前浏览别人的博客网站,打开外链时会有一个等待时间的代码,虽然不知道有什么用,但觉的挺有档次..今天正好看到教程,就自己也加上了,就复制粘贴些代码可以了 首先创建一个php文件,名字随便,如果你不想改 ...
- [Swift]LeetCode1025. 除数博弈 | Divisor Game
Alice and Bob take turns playing a game, with Alice starting first. Initially, there is a number N o ...
- 浏览器与服务端请求响应流程与HTTP协议
浏览器与服务端请求响应流程图: 1.HTTP概要 1.1. 定义 HTTP(HyperText Transfer Protocol,超文本传输协议)最早就是计算机与计算机之间沟通的一种标准协议,这种 ...
- Spring Data Jpa接口简介
Repository接口 public interface Repository<T, ID> {....} 提供了按方法名称的查询方式: 提供了@Query的查询方式 可能遇到的错误: ...
- 如何创建测试程序调试nginx数据结构
如何创建测试程序调试nginx数据结构 由于在学习nginx的过程中遇到很多数据结构,往往我都想写一个程序来跑一下,看下到底返回什么.最开始想的方法是使用nginx make 完成之后的.o文件,做L ...
- C# 之 Socket 简单入门示例
这个例子只是简单实现了如何使用 Socket 类实现面向连接的通信. 注意:此例子的目的只是为了说明用套接字写程序的大概思路,而不是实际项目中的使用程序.在这个例子中,实际上还有很多问题没有解决,如消 ...