PHP反射实际利用示范_php本领_脚本之家金沙8331网址

正文实例汇报了PHP反射实际使用。分享给大家供大家参谋,具体如下:

本文实例陈说了Laravel达成结构函数自动依赖注入的措施。分享给我们供大家参照他事他说加以考察,具体如下:

1.自动生成文书档案

PHP反射实际利用示范_php本领_脚本之家金沙8331网址。在Laravel的构造函数中得以完结自动信赖注入,而无需实例化从前先实例化需求的类,如代码所示:

凭借反射的分析类,接口,函数和方法的内部布局,方法和函数的参数,甚至类的属性和办法,能够自动生成文书档案。

threads = $threads; $this->tags = $tags; $this->threadCreator = $threadCreator; $this->replies = $replies; }}
id; } public function setId { $this->id = $id; }}$ref = new ReflectionClass;$doc = $ref->getDocComment();echo $ref->getName() . ':' . getComment , "
";echo "属性列表:
";printf("%-15s%-10s%-40s
", 'Name', 'Access', 'Comment');$attr = $ref->getProperties();foreach  { printf("%-15s%-10s%-40s
", $row->getName, getComment;}echo "常量列表:
";printf("%-15s%-10s
", 'Name', 'Value');$const = $ref->getConstants();foreach ($const as $key => $val) { printf("%-15s%-10s
", $key, $val);}echo "

";echo "方法列表
";printf("%-15s%-10s%-30s%-40s
", 'Name', 'Access', 'Params', 'Comment');$methods = $ref->getMethods();foreach  { printf("%-15s%-10s%-30s%-40s
", $row->getName, getParams, getComment;}// 获取权限function getAccess{ if  { return 'Public'; } if ($method->isProtected { return 'Protected'; } if  { return 'Private'; }}// 获取方法参数信息function getParams{ $str = ''; $parameters = $method->getParameters(); foreach  { $str .= $row->getName() . ','; if ($row->isDefaultValueAvailable { $str .= "Default: {$row->getDefaultValue()}"; } } return $str ? $str : '';}// 获取注释function getComment{ $comment = $var->getDocComment(); // 简单的获取了第一行的信息,这里可以自行扩展 preg_match *?/', $comment, $res); return isset ? $res[1] : '';}

只顾布局函数中的多少个连串限定,其实并未地点实例化这些Controller并把那多少个类其他参数字传送进去,Laravel会自动检测类的布局函数中的类型节制参数,并自动识别是或不是开头化并传到。

Student:属性列表:Name Access Commentid Public 客户ID常量列表:Name
ValueNORMAL 1FORBIDDEN 2主意列表Name Access Params CommentgetId Public
获取idsetId Public id,Default: 1

源码vendor/illuminate/container/Container.php中的build方法:

2.实现 MVC 架构

$constructor = $reflector->getConstructor;

至今广大框架都是 MVC 的构造,依照路由音信定位调整器 和章程的名号,之后选取反射达成全自动调用。

那边会解析类的构造函数,在这里边打字与印刷看:

$class = new ReflectionClass . 'Controller');$controller = $class->newInstance();if ($class->hasMethod { $method = $class->getMethod; $method->invokeArgs($controller, $arguments);} else { throw new Exception("{$controller} controller method {$method} not exists!");}

它会搜索布局函数的参数,再看完整的build方法实行的操作:

3.达成单元测量试验

public function build($concrete, array $parameters = []){ // If the concrete type is actually a Closure, we will just execute it and // hand back the results of the functions, which allows functions to be // used as resolvers for more fine-tuned resolution of these objects. if ($concrete instanceof Closure) { return $concrete; } $reflector = new ReflectionClass; // If the type is not instantiable, the developer is attempting to resolve // an abstract type such as an Interface of Abstract Class and there is // no binding registered for the abstractions so we need to bail out. if (! $reflector->isInstantiable { $message = "Target [$concrete] is not instantiable."; throw new BindingResolutionContractException; } $this->buildStack[] = $concrete; $constructor = $reflector->getConstructor(); // If there are no constructors, that means there are no dependencies then // we can just resolve the instances of the objects right away, without // resolving any other types or dependencies out of these containers. if ) { array_pop; return new $concrete; } $dependencies = $constructor->getParameters(); // Once we have all the constructor's parameters we can create each of the // dependency instances and then use the reflection instances to make a // new instance of this class, injecting the created dependencies in. $parameters = $this->keyParametersByArgument( $dependencies, $parameters ); $instances = $this->getDependencies( $dependencies, $parameters ); array_pop; return $reflector->newInstanceArgs;}

诚如意况下大家会对函数和类实行测量试验,剖断其是还是不是能够按大家预料再次回到结果,大家能够用反射达成一个精短通用的类测量检验用例。

现实从容器中拿走实例的法门:

hasMethod { $method = $ref->getMethod; $res = $method->invokeArgs; if{ echo "测试结果正确"; }; }}testEqual('Calc@plus', 3, [1, 2]);echo "
";testEqual('Calc@minus', -1, [1, 2]);
protected function resolveClass(ReflectionParameter $parameter){ try { return $this->make($parameter->getClass; } // If we can not resolve the class instance, we will check to see if the value // is optional, and if it is we will return the optional parameter value as // the value of the dependency, similarly to how we do this with scalars. catch (BindingResolutionContractException $e) { if ($parameter->isOptional { return $parameter->getDefaultValue(); } throw $e; }}

那是类的测量试验方法,也得以行使反射完毕函数的测量检验方法。

框架底层通过Reflection反射为付出节省了许多细节,达成了自动信任注入。这里不做持续深切研商了。

invokeArgs;?>

写了叁个模拟那么些进程的类测量试验:

此地只是我大致写的五个测量试验用例,PHPUnit 单元测量检验框架比比较大程度上信赖了
Reflection 的本性,能够通晓下。

kulou = $kulou; $this->junjun = $junjun; }}//$tanteng = new tanteng,new junjun;$reflector = new ReflectionClass;$constructor = $reflector->getConstructor();$dependencies = $constructor->getParameters();print_r;exit;

4.十分 DI 容器解除看重

规律是透过ReflectionClass类解析类的布局函数,並且收取布局函数的参数,进而判定信赖关系,从容器中取,并自动注入。

Laravel 等超多框架都以应用 Reflection 化解信赖注入难点,具体可查阅
Laravel 源码实行分析。

转自:小谈博客

上边大家代码轻松完毕多少个 DI 容器演示 Reflection 消除信任注入难点。

越来越多关于Laravel相关内容感兴趣的读者可查看本站专项论题:《Laravel框架入门与进级教程》、《php特出开采框架总括》、《smarty模板入门底蕴教程》、《php日期与时光用法计算》、《php面向对象程序设计入门教程》、《php字符串用法计算》、《php+mysql数据库操作入门教程》及《php无动于中数据库操作技艺汇总》

bulid; } // 获取实例 public function bulid { // 如果是匿名函数,直接执行,并返回结果 if ($className instanceof Closure) { return $className; } // 已经是实例化对象的话,直接返回 if) { return $className; } // 如果是类的话,使用反射加载 $ref = new ReflectionClass; // 监测类是否可实例化 if (!$ref->isInstantiable { throw new Exception('class' . $className . ' not find'); } // 获取构造函数 $construtor = $ref->getConstructor(); // 无构造函数,直接实例化返回 if  { return new $className; } // 获取构造函数参数 $params = $construtor->getParameters(); // 解析构造函数 $dependencies = $this->getDependecies; // 创建新实例 return $ref->newInstanceArgs; } // 分析参数,如果参数中出现依赖类,递归实例化 public function getDependecies { $data = []; foreach { $tmp = $param->getClass { $data[] = $this->setDefault; } else { $data[] = $this->bulid; } } return $data; } // 设置默认值 public function setDefault { if ($param->isDefaultValueAvailable { return $param->getDefaultValue(); } throw new Exception; }}class Demo{ public function __construct { echo $calc->plus; }}class Calc{ public function plus { return $a + $b; } public function minus { return $a - $b; }}$di = new DI();$di->calc = 'Calc';$di->demo = 'Demo';$di->demo;//输出结果为3

期望本文所述对我们基于Laravel框架的PHP程序设计具备利于。

越多关于PHP相关内容感兴趣的读者可查阅本站专项论题:《php面向对象程序设计入门教程》、《PHP数组操作技艺大全》、《PHP基本语法入门教程》、《PHP运算与运算符用法总括》、《php字符串用法计算》、《php+mysql数据库操作入门教程》及《php多如牛毛数据库操作技艺汇总》

可望本文所述对大家PHP程序设计有所支持。

发表评论

电子邮件地址不会被公开。 必填项已用*标注