反射,直观理解就是根据到达地找到出发地和来源。好比,一个光秃秃的对象,咱们能够仅仅经过这个对象就能知道它所属的类、拥有哪些方法。php
反射是指在PHP运行状态中,扩展分析PHP程序,导出或提出关于类、方法、属性、参数等的详细信息,包括注释。这种动态获取信息以及动态调用对象方法的功能称为反射API。mysql
ReflectionClass: 获取类声明时的结构sql
ReflectionObject: 可获取类实例化后的结构数据库
代码以下:
class test{
private $name;
private $sex;
function __construct(){
$this->aaa='aaa';
}
}
$test=new test();
$reflect=new ReflectionClass($test);
$pro=$reflect->getDefaultProperties();
print_r($pro);//打印结果:Array ( [name] => [sex] => )
echo $test->aaa;//打印结果:aaa数组
在这个test类中,声明了两个成员变量$name和$sex,可是在构造函数中,又声明了一个变量$aaa,初始化类,使用反射类打印默认成员属性只有声明的两个成员变量属性,可是打印类的$aaa变量发现仍是能够输出结果。
请问类的成员变量不用声明,在函数中声明也是能够的吗,有什么区别?
在你这个例子中,使用ReflectionClass是不恰当的,由于__construct只有在实例化class时,才会执行。
也就是说ReflectionClass更多的是反射类声明时的结构,而不是类实例化后的结构,因此没有输出属性aaa是正确,由于属性aaa确实是(在类声明时)不存在的。
那么怎么看属性aaa呢,应该用ReflectionObject反射实例化后的结构,例如函数
代码以下:
<?php
class test{
private $name;
private $sex;
function __construct(){
$this->aaa='aaa';
}
}
$test=new test();
$reflect=new ReflectionObject($test);
$pro=$reflect->getProperties();
print_r($pro);this
通过实例化之后,属性aaa才会存在,这时你就能看到属性aaa了spa
例子:插件
1代理 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 |
|
如今,要获取这个student对象的方法和属性列表该怎么作呢?如如下代码所示:
1 2 3 4 5 6 7 8 9 10 11 |
|
也能够不用反射API,使用class函数,返回对象属性的关联数组以及更多的信息:
1 2 3 4 5 6 |
|
假如这个对象是从其余页面传过来的,怎么知道它属于哪一个类呢?一句代码就能够搞定:
1 2 |
|
反射API的功能显然更强大,甚至能还原这个类的原型,包括方法的访问权限等,如:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 |
|
输出以下:
1 2 3 4 5 6 7 8 |
|
不只如此,PHP手册中关于反射API更是有几十个,能够说,反射完整地描述了一个类或者对象的原型。反射不只能够用于类和对象,还能够用于函数、扩展模块、异常等。
反射能够用于文档生成。所以能够用它对文件里的类进行扫描,逐个生成描述文档。
既然反射能够探知类的内部结构,那么是否是能够用它作hook实现插件功能呢?或者是作动态代理呢?
例如:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 |
|
在日常开发中,用到反射的地方很少:一个是对对象进行调试,另外一个是获取类的信息。在MVC和插件开发中,使用反射很常见,可是反射的消耗也很大,在能够找到替代方案的状况下,就不要滥用。
PHP有Token函数,能够经过这个机制实现一些反射功能。从简单灵活的角度讲,使用已经提供的反射API是可取的。
不少时候,善用反射能保持代码的优雅和简洁,但反射也会破坏类的封装性,由于反射可使本不该该暴露的方法或属性被强制暴露了出来,这既是优势也是缺点。