欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

PHP反射实际应用示例

程序员文章站 2022-06-14 13:44:24
本文实例讲述了php反射实际应用。分享给大家供大家参考,具体如下: 1.自动生成文档 根据反射的分析类,接口,函数和方法的内部结构,方法和函数的参数,以及类的属性和方法...

本文实例讲述了php反射实际应用。分享给大家供大家参考,具体如下:

1.自动生成文档

根据反射的分析类,接口,函数和方法的内部结构,方法和函数的参数,以及类的属性和方法,可以自动生成文档。

<?php
class student
{
  const normal = 1;
  const forbidden = 2;
  /**
   * 用户id
   * @var 类型
   */
  public $id;
  /**
   * 获取id
   * @return int
   */
  public function getid()
  {
    return $this->id;
  }
  public function setid($id = 1)
  {
    $this->id = $id;
  }
}
$ref = new reflectionclass('student');
$doc = $ref->getdoccomment();
echo $ref->getname() . ':' . getcomment($ref) , "<br/>";
echo "属性列表:<br/>";
printf("%-15s%-10s%-40s<br/>", 'name', 'access', 'comment');
$attr = $ref->getproperties();
foreach ($attr as $row) {
  printf("%-15s%-10s%-40s<br/>", $row->getname(), getaccess($row), getcomment($row));
}
echo "常量列表:<br/>";
printf("%-15s%-10s<br/>", 'name', 'value');
$const = $ref->getconstants();
foreach ($const as $key => $val) {
  printf("%-15s%-10s<br/>", $key, $val);
}
echo "<br/><br/>";
echo "方法列表<br/>";
printf("%-15s%-10s%-30s%-40s<br/>", 'name', 'access', 'params', 'comment');
$methods = $ref->getmethods();
foreach ($methods as $row) {
  printf("%-15s%-10s%-30s%-40s<br/>", $row->getname(), getaccess($row), getparams($row), getcomment($row));
}
// 获取权限
function getaccess($method)
{
  if ($method->ispublic()) {
    return 'public';
  }
  if ($method->isprotected()) {
    return 'protected';
  }
  if ($method->isprivate()) {
    return 'private';
  }
}
// 获取方法参数信息
function getparams($method)
{
  $str = '';
  $parameters = $method->getparameters();
  foreach ($parameters as $row) {
    $str .= $row->getname() . ',';
    if ($row->isdefaultvalueavailable()) {
      $str .= "default: {$row->getdefaultvalue()}";
    }
  }
  return $str ? $str : '';
}
// 获取注释
function getcomment($var)
{
  $comment = $var->getdoccomment();
  // 简单的获取了第一行的信息,这里可以自行扩展
  preg_match('/\* (.*) *?/', $comment, $res);
  return isset($res[1]) ? $res[1] : '';
}

输出结果:

student:
属性列表:
name access comment
id public 用户id
常量列表:
name value
normal 1
forbidden 2
方法列表
name access params comment
getid public 获取id
setid public id,default: 1

2.实现 mvc 架构

现在好多框架都是 mvc 的架构,根据路由信息定位控制器($controller) 和方法($method) 的名称,之后使用反射实现自动调用。

$class = new reflectionclass(ucfirst($controller) . 'controller');
$controller = $class->newinstance();
if ($class->hasmethod($method)) {
  $method = $class->getmethod($method);
  $method->invokeargs($controller, $arguments);
} else {
  throw new exception("{$controller} controller method {$method} not exists!");
}

3.实现单元测试

一般情况下我们会对函数和类进行测试,判断其是否能够按我们预期返回结果,我们可以用反射实现一个简单通用的类测试用例。

<?php
class calc
{
  public function plus($a, $b)
  {
    return $a + $b;
  }
  public function minus($a, $b)
  {
    return $a - $b;
  }
}
function testequal($method, $assert, $data)
{
  $arr = explode('@', $method);
  $class = $arr[0];
  $method = $arr[1];
  $ref = new reflectionclass($class);
  if ($ref->hasmethod($method)) {
    $method = $ref->getmethod($method);
    $res = $method->invokeargs(new $class, $data);
    if($res === $assert){
      echo "测试结果正确";
    };
  }
}
testequal('calc@plus', 3, [1, 2]);
echo "<br/>";
testequal('calc@minus', -1, [1, 2]);

这是类的测试方法,也可以利用反射实现函数的测试方法。

<?php
function title($title, $name)
{
  return sprintf("%s. %s\r\n", $title, $name);
}
$function = new reflectionfunction('title');
echo $function->invokeargs(array('dr', 'phil'));
?>

这里只是我简单写的一个测试用例,phpunit 单元测试框架很大程度上依赖了 reflection 的特性,可以了解下。

4.配合 di 容器解决依赖

laravel 等许多框架都是使用 reflection 解决依赖注入问题,具体可查看 laravel 源码进行分析。

下面我们代码简单实现一个 di 容器演示 reflection 解决依赖注入问题。

<?php
class di
{
  protected static $data = [];
  public function __set($k, $v)
  {
    self::$data[$k] = $v;
  }
  public function __get($k)
  {
    return $this->bulid(self::$data[$k]);
  }
  // 获取实例
  public function bulid($classname)
  {
    // 如果是匿名函数,直接执行,并返回结果
    if ($classname instanceof closure) {
      return $classname($this);
    }
    // 已经是实例化对象的话,直接返回
    if(is_object($classname)) {
      return $classname;
    }
    // 如果是类的话,使用反射加载
    $ref = new reflectionclass($classname);
    // 监测类是否可实例化
    if (!$ref->isinstantiable()) {
      throw new exception('class' . $classname . ' not find');
    }
    // 获取构造函数
    $construtor = $ref->getconstructor();
    // 无构造函数,直接实例化返回
    if (is_null($construtor)) {
      return new $classname;
    }
    // 获取构造函数参数
    $params = $construtor->getparameters();
    // 解析构造函数
    $dependencies = $this->getdependecies($params);
    // 创建新实例
    return $ref->newinstanceargs($dependencies);
  }
  // 分析参数,如果参数中出现依赖类,递归实例化
  public function getdependecies($params)
  {
    $data = [];
    foreach($params as $param)
    {
      $tmp = $param->getclass();
      if (is_null($tmp)) {
        $data[] = $this->setdefault($param);
      } else {
        $data[] = $this->bulid($tmp->name);
      }
    }
    return $data;
  }
  // 设置默认值
  public function setdefault($param)
  {
    if ($param->isdefaultvalueavailable()) {
      return $param->getdefaultvalue();
    }
    throw new exception('no default value!');
  }
}
class demo
{
  public function __construct(calc $calc)
  {
    echo $calc->plus(1, 2);
  }
}
class calc
{
  public function plus($a, $b)
  {
    return $a + $b;
  }
  public function minus($a, $b)
  {
    return $a - $b;
  }
}
$di = new di();
$di->calc = 'calc';
$di->demo = 'demo';
$di->demo;//输出结果为3

更多关于php相关内容感兴趣的读者可查看本站专题:《php面向对象程序设计入门教程》、《php数组(array)操作技巧大全》、《php基本语法入门教程》、《php运算与运算符用法总结》、《php字符串(string)用法总结》、《php+mysql数据库操作入门教程》及《php常见数据库操作技巧汇总

希望本文所述对大家php程序设计有所帮助。