设为首页 - 加入收藏 ASP站长网(Aspzz.Cn)- 科技、建站、经验、云计算、5G、大数据,站长网!
热搜: 创业者 手机 数据
当前位置: 首页 > 站长学院 > PHP教程 > 正文

PHP反射原理与用法深入分析

发布时间:2022-06-25 14:03 所属栏目:121 来源:互联网
导读:这篇文章主要介绍了PHP反射原理与用法,结合实例形式深入分析了PHP反射的概念、原理、应用场景及相关操作技巧,需要的朋友可以参考下。 本文实例讲述了PHP反射原理与用法,分享给大家供大家参考,具体如下: 说到反射,实际上包含两个概念: 检视 introspectio
  这篇文章主要介绍了PHP反射原理与用法,结合实例形式深入分析了PHP反射的概念、原理、应用场景及相关操作技巧,需要的朋友可以参考下。
 
  本文实例讲述了PHP反射原理与用法,分享给大家供大家参考,具体如下:
 
  说到反射,实际上包含两个概念:
 
  检视 introspection 判断类、方法是否存在,父子类关系,调用关系等,检视的函数文档
 
  反射 Reflection 获取类里的方法、属性,注释等,反射类的文档
 
  PHP官方文档写得很清晰了,下面我就说一下具体的应用。
 
  1.参数检测
 
  有时候需要在函数里需要判断传入的参数类型是否合法。
 
  这时可以使用is_a、is_subclass_of来检测。或者结合反射,做更多检测。
 
  2.动态调用
 
  在依赖注入中,常见到这种用法,比如Laravel5.5中的Container.php
 
  public function build($concrete)
    {
      // 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($this, $this->getLastParameterOverride());
      }
      $reflector = new ReflectionClass($concrete);
      // 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()) {
        return $this->notInstantiable($concrete);
      }
      $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 (is_null($constructor)) {
        array_pop($this->buildStack);
        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.
      $instances = $this->resolveDependencies(
        $dependencies
      );
      array_pop($this->buildStack);
      return $reflector->newInstanceArgs($instances);
    }
  上述代码先判断是否是闭包,如果是,直接返回。不是则通过new ReflectionClass($concrete);
 
  生成反射类的实例,然后获取这个类的构造函数和参数,进行初始化的过程。
 
  注意
 
  反射里一个比较重要的用法invoke
 
  当已知这个类的时候,可以通过构造ReflectionMethod来直接调用,如:
 
  class HelloWorld {
  
    public function sayHelloTo($name) {
      return 'Hello ' . $name;
    }
  
  }
  
  $reflectionMethod = new ReflectionMethod('HelloWorld', 'sayHelloTo');
  echo $reflectionMethod->invoke(new HelloWorld(), 'Mike');
  当不知道这个类时,知道类的对象,可以用ReflectionObject获取ReflectionMethod后调用,如:
 
  class HelloWorld {
  
    public function sayHelloTo($name) {
      return 'Hello ' . $name;
    }
  
  }
  
  $hello = new HelloWorld();
  
  $refObj = new ReflectionObject($hello);
  $refMethod = $refObj->getMethod('sayHelloTo');
  echo $refMethod->invoke($hello,'Mike');
  调用流程一般就是获取反射类ReflectionClass/反射对象ReflectionObject的实例,然后获取ReflectionMethod后,invoke。
 
  3.获取注释,生成文档
 
  比如PHPDoc
 
  4.注解,增强版的注释,符合一定的规则
 
  比如某些框架的路由,便是通过注解实现的。
 
  5.不要为了反射而反射
 
  PHP是一门动态语言,其实可以直接通过字符串来调用类或函数,如下:
 
  class HelloWorld {
    public function sayHelloTo($name) {
      return 'Hello ' . $name;
    }
  }
  $hello = 'HelloWorld';
  $helloSay = 'sayHelloTo';
  $helloIntance = new $hello;
  echo $helloIntance->$helloSay('Mike');
  那么为什么还需要反射呢?
 
  功能更强大
 
  更安全,防止直接调用没有暴露的内部方法
 
  可维护,直接写字符串是硬编码。

(编辑:ASP站长网)

    网友评论
    推荐文章
      热点阅读