反射
面向对象编程中对象被赋予了自省的能力,而这个自省的过程就是反射。
反射,直观理解就是根据到达地找到出发地和来源。比如,一个光秃秃的对象,我们可以仅仅通过这个对象就能知道它所属的类、拥有哪些方法。
反射是指在PHP运行状态中,扩展分析PHP程序,导出或提出关于类、方法、属性、参数等的详细信息,包括注释。这种动态获取信息以及动态调用对象方法的功能称为反射API。
// 获取对象属性列表
$reflect = new ReflectionObject($obj);
$props = $reflect->getProperties();
$Methods = $Properties = array();
foreach ($props as $prop) {
print $prop->getName() . "\n";
$Properties[$prop->getName()] = $prop;
// $prop->isPublic();
// $prop->isPrivate();
// $prop->isProtected();
// $prop->isStatic();
}
// 获取对象方法列表
$m = $reflect->getMethods();
foreach ($m as $prop) {
print $prop->getName() . "\n";
}
也可以不用反射API,使用class函数,返回对象属性的关联数组以及更多的信息:
// 返回对象属性的关联数组
var_dump(get_object_vars($obj));
// 类属性
var_dump(get_class_vars(get_class($obj)));
// 返回由类的方法名组成的数组
var_dump(get_class_methods(get_class($obj)));
反射不仅可以用于类和对象,还可以用于函数、扩展模块、异常等。
反射有什么作用?
反射可以用于文档生成。因此可以用它对文件里的类进行扫描,逐个生成描述文档。
既然反射可以探知类的内部结构,那么是不是可以用它做hook实现插件功能呢?或者是做动态代理呢?
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');