面向工具编程中工具被授予了自省的能力,而这个自省的过程便是反射

反射,直不雅观理解便是根据到达地找到出发地和来源。
比如,一个光秃秃的工具,我们可以仅仅通过这个工具就能知道它所属的类、拥有哪些方法。

反射是指在PHP运行状态中,扩展剖析PHP程序,导出或提出关于类、方法、属性、参数等的详细信息,包括注释。
这种动态获取信息以及动态调用工具方法的功能称为反射API。

php反射工厂模式php反射机制用法详解 Ruby

如何利用反射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=\"大众正在为你设置默认值\"大众;

}

return $this->$name;

}

}

$student=new person();

$student->name='Tom';

$student->gender='male';

$student->age=24;

现在,要获取这个student工具的方法和属性列表该怎么做呢?如以下代码所示:

// 获取工具属性列表

$reflect = new ReflectionObject($student);

$props = $reflect->getProperties();

foreach ($props as $prop) {

print $prop->getName() .\"大众\n\公众;

}

// 获取工具方法列表

$m=$reflect->getMethods();

foreach ($m as $prop) {

print $prop->getName() .\公众\n\"大众;

}

也可以不用反射API,利用class函数,返回工具属性的关联数组以及更多的信息:

// 返回工具属性的关联数组

var_dump(get_object_vars($student));

// 类属性

var_dump(get_class_vars(get_class($student)));

// 返回由类的方法名组成的数组

var_dump(get_class_methods(get_class($student)));

如果这个工具是从其他页面传过来的,怎么知道它属于哪个类呢?一句代码就可以搞定:

// 获取工具属性列表所属的类

echo get_class($student);

反射API的功能显然更强大,乃至能还原这个类的原型,包括方法的访问权限等,如:

// 反射获取类的原型

$obj = new ReflectionClass('person');

$className = $obj->getName();

$Methods = $Properties = array();

foreach($obj->getProperties() as $v)

{

    $Properties[$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

{

public gender

public name

function get(){}

function set(){}

function say(){}

}

不仅如此,PHP手册中关于反射API更是有几十个,可以说,反射完全地描述了一个类或者工具的原型。
反射不仅可以用于类和工具,还可以用于函数、扩展模块、非常等。

反射有什么浸染?

反射可以用于文档天生。
因此可以用它对文件里的类进行扫描,逐个天生描述文档。

既然反射可以探知类的内部构造,那么是不是可以用它做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和插件开拓中,利用反射很常见,但是反射的花费也很大,在可以找到替代方案的情形下,就不要滥用。

很多时候,善用反射能保持代码的优雅和简洁,但反射也会毁坏类的封装性,由于反射可以使本不应该暴露的方法或属性被逼迫暴露了出来,这既是优点也是缺陷。