Maison  >  Article  >  développement back-end  >  Explication détaillée de l'utilisation de la réflexion en PHP

Explication détaillée de l'utilisation de la réflexion en PHP

coldplay.xixi
coldplay.xixiavant
2020-06-16 17:46:292510parcourir

Explication détaillée de l'utilisation de la réflexion en PHP

Parlons de l'application de la réflexion dans le développement réel.

  • Générer automatiquement des documents
  • Implémenter l'architecture MVC
  • Implémenter les tests unitaires
  • Travailler avec le conteneur DI pour résoudre les dépendances

Générer automatiquement la documentation

Les documents peuvent être générés automatiquement en fonction de l'analyse réfléchie des classes, des interfaces, des structures internes des fonctions et des méthodes, des paramètres des méthodes et des fonctions, ainsi que des attributs et des méthodes. de cours.

/**
 * 学生类
 *
 * 描述信息
 */
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) , "\n";
echo "属性列表:\n";
printf("%-15s%-10s%-40s\n", 'Name', 'Access', 'Comment');
$attr = $ref->getProperties();
foreach ($attr as $row) {
    printf("%-15s%-10s%-40s\n", $row->getName(), getAccess($row), getComment($row));
}
echo "常量列表:\n";
printf("%-15s%-10s\n", 'Name', 'Value');
$const = $ref->getConstants();
foreach ($const as $key => $val) {
    printf("%-15s%-10s\n", $key, $val);
}
echo "\n\n";
echo "方法列表\n";
printf("%-15s%-10s%-30s%-40s\n", 'Name', 'Access', 'Params', 'Comment');
$methods = $ref->getMethods();
foreach ($methods as $row) {
    printf("%-15s%-10s%-30s%-40s\n", $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] : '';
}
​ ​ ​

Exécutez php file.php pour voir les informations du document correspondant.

Implémentation de l'architecture MVC

De nombreux frameworks utilisent désormais l'architecture MVC Ils localisent les noms des contrôleurs ($controller) et des méthodes ($method) en fonction des informations de routage, puis utilisent la réflexion. pour mettre en œuvre un appel automatique.

$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!");
}

Explication détaillée de l'utilisation de la réflexion en PHP

Implémentation des tests unitaires

Généralement, nous testerons les fonctions et les classes pour déterminer si elles peuvent renvoyer les résultats que nous pouvons utiliser. réflexion pour implémenter un cas de test de classe simple et général.

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);
        var_dump($res === $assert);
    }
}
testEqual('Calc@plus', 3, [1, 2]);
testEqual('Calc@minus', -1, [1, 2]);
​ ​ ​

Il s'agit de la méthode de test de la classe, et vous pouvez également utiliser la réflexion pour implémenter la méthode de test de la fonction.
Ceci est juste un cas de test que j'ai simplement écrit PHPUnit Le framework de tests unitaires s'appuie fortement sur les fonctionnalités de Reflection, vous pouvez en apprendre davantage.

Fonctionne avec les conteneurs DI pour résoudre les dépendances

Laravel et de nombreux autres frameworks utilisent Reflection pour résoudre les problèmes d'injection de dépendances. Pour plus de détails, veuillez consulter le code source Laravel pour analyse.
Ci-dessous, notre code implémente simplement une DI démonstration de conteneur Reflection pour résoudre le problème d'injection de dépendances.

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);
    }
}
$di = new DI();
$di->calc = 'Calc'; // 加载单元测试用例中 Calc 类
$di->demo = 'Demo';
$di->demo;

Faites attention à l'ordre de et calc ci-dessus, et il ne peut pas être inversé, sinon une erreur sera signalée car demo dépendance Demo, tout d'abord, la dépendance doit être définie. Calc Lors de l'instanciation de
, la classe Demo sera utilisée, ce qui signifie que Calc dépend de Demo, mais si elle n'est pas trouvée sur Calc, une erreur sera générée, donc d'abord À définir                                                                     $data. $di->calc = 'Calc'

La réflexion est une fonctionnalité très intéressante, utilisez-la, mais n'en abusez pas.

FinInsistez pour partager la technologie originale, votre soutien m'encouragera à continuer

Tutoriel recommandé : "

tutoriel php"

Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

Déclaration:
Cet article est reproduit dans:. en cas de violation, veuillez contacter admin@php.cn Supprimer