您好,欢迎访问一九零五行业门户网

详解PHP的反射使用

下面我们讲下反射在实际开发中的应用。
自动生成文档实现 mvc 架构实现单元测试配合 di 容器解决依赖…自动生成文档根据反射的分析类,接口,函数和方法的内部结构,方法和函数的参数,以及类的属性和方法,可以自动生成文档。
/** * 学生类 * * 描述信息 */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] : '';}
运行 php file.php 就可以看到相应的文档信息。
实现 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!);}
实现单元测试一般情况下我们会对函数和类进行测试,判断其是否能够按我们预期返回结果,我们可以用反射实现一个简单通用的类测试用例。
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]);
这是类的测试方法,也可以利用反射实现函数的测试方法。
这里只是我简单写的一个测试用例,phpunit 单元测试框架很大程度上依赖了 reflection 的特性,可以了解下。
配合 di 容器解决依赖laravel 等许多框架都是使用 reflection 解决依赖注入问题,具体可查看 laravel 源码进行分析。
下面我们代码简单实现一个 di 容器演示 reflection 解决依赖注入问题。
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;
注意上面的 calc 和 demo 的顺序,不能颠倒,不然的话会报错,原因是由于 demo 依赖 calc,首先要定义依赖关系。
在 demo 实例化的时候,会用到 calc 类,也就是说 demo 依赖于 calc,但是在 $data 上面找不到的话,会抛出错误,所以首先要定义            $di->calc = 'calc'。
reflection 是一个非常 cool 的功能,使用它,但不要滥用它。
end
坚持原创技术分享,您的支持将鼓励我继续推荐教程:《php教程》
以上就是详解php的反射使用的详细内容。
其它类似信息

推荐信息