PHP反射机制原理与用法详解
本文实例讲述了PHP反射机制原理与用法。分享给大家供大家参考,具体如下:
反射
面向对象编程中对象被赋予了自省的能力,而这个自省的过程就是反射。
反射,直观理解就是根据到达地找到出发地和来源。比如,一个光秃秃的对象,我们可以仅仅通过这个对象就能知道它所属的类、拥有哪些方法。
反射是指在PHP运行状态中,扩展分析PHP程序,导出或提出关于类、方法、属性、参数等的详细信息,包括注释。这种动态获取信息以及动态调用对象方法的功能称为反射API。
如何使用反射API
<?php classperson{ public$name; public$gender; publicfunctionsay(){ echo$this->name,"\tis",$this->gender,"\r\n"; } publicfunctionset($name,$value){ echo"Setting$nameto$value\r\n"; $this->$name=$value; } publicfunctionget($name){ if(!isset($this->$name)){ echo'未设置'; $this->$name="正在为你设置默认值"; } return$this->$name; } } $student=newperson(); $student->name='Tom'; $student->gender='male'; $student->age=24;
现在,要获取这个student对象的方法和属性列表该怎么做呢?如以下代码所示:
//获取对象属性列表 $reflect=newReflectionObject($student); $props=$reflect->getProperties(); foreach($propsas$prop){ print$prop->getName()."\n"; } //获取对象方法列表 $m=$reflect->getMethods(); foreach($mas$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)));
假如这个对象是从其他页面传过来的,怎么知道它属于哪个类呢?一句代码就可以搞定:
//获取对象属性列表所属的类 echoget_class($student);
反射API的功能显然更强大,甚至能还原这个类的原型,包括方法的访问权限等,如:
//反射获取类的原型 $obj=newReflectionClass('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($Propertiesas$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($Methodsas$k=>$v) { echo"\tfunction{$k}(){}\n"; } echo"}\n";
输出如下:
classperson { publicgender publicname functionget(){} functionset(){} functionsay(){} }
不仅如此,PHP手册中关于反射API更是有几十个,可以说,反射完整地描述了一个类或者对象的原型。反射不仅可以用于类和对象,还可以用于函数、扩展模块、异常等。
反射有什么作用
反射可以用于文档生成。因此可以用它对文件里的类进行扫描,逐个生成描述文档。
既然反射可以探知类的内部结构,那么是不是可以用它做hook实现插件功能呢?或者是做动态代理呢?
例如:
<?php classmysql{ functionconnect($db){ echo"连接到数据库${db[0]}\r\n"; } } classsqlproxy{ private$target; functionconstruct($tar){ $this->target[]=new$tar(); } functioncall($name,$args){ foreach($this->targetas$obj){ $r=newReflectionClass($obj); if($method=$r->getMethod($name)){ if($method->isPublic()&&!$method->isAbstract()){ echo"方法前拦截记录LOG\r\n"; $method->invoke($obj,$args); echo"方法后拦截\r\n"; } } } } } $obj=newsqlproxy('mysql'); $obj->connect('member');
在平常开发中,用到反射的地方不多:一个是对对象进行调试,另一个是获取类的信息。在MVC和插件开发中,使用反射很常见,但是反射的消耗也很大,在可以找到替代方案的情况下,就不要滥用。
PHP有Token函数,可以通过这个机制实现一些反射功能。从简单灵活的角度讲,使用已经提供的反射API是可取的。
很多时候,善用反射能保持代码的优雅和简洁,但反射也会破坏类的封装性,因为反射可以使本不应该暴露的方法或属性被强制暴露了出来,这既是优点也是缺点。
更多关于PHP相关内容感兴趣的读者可查看本站专题:《php面向对象程序设计入门教程》、《PHP基本语法入门教程》、《PHP运算与运算符用法总结》、《PHP网络编程技巧总结》、《PHP数组(Array)操作技巧大全》、《php字符串(string)用法总结》、《php+mysql数据库操作入门教程》及《php常见数据库操作技巧汇总》
希望本文所述对大家PHP程序设计有所帮助。