最新公告
  • 欢迎您访问爱上源码网,分享精品整站源码,网站模板,游戏源码,APP小程序源码以及视频教程免费下载;服务永无止境!立即加入我们
  • 详解PHP的反射使用

    爱上源码网文章详解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!");
    }

    爱上源码网文章详解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);
            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;

    注意上面的 calcdemo 的顺序,不能颠倒,不然的话会报错,原因是由于 Demo 依赖 Calc,首先要定义依赖关系。
    Demo 实例化的时候,会用到 Calc 类,也就是说 Demo 依赖于 Calc,但是在 $data 上面找不到的话,会抛出错误,所以首先要定义 $di->calc = 'Calc'

    Reflection 是一个非常 Cool 的功能,使用它,但不要滥用它。

    End

    坚持原创技术分享,您的支持将鼓励我继续

    推荐教程:《php教程》

    以上就是详解PHP的反射使用的详细内容,更多请关注爱上源码网其它相关文章!

  • 微信
  • 分享
  • 相关标签:PHP
  • 本文转载于:juejin.im,如有侵犯,请联系916990011@qq.com删除
    • 上一篇:PHP实战之Redis常见7种使用场景
    • 下一篇:最全的php数组操作方法汇总,让你轻松掌握!

    相关文章

    相关视频

    • phpmyadmin配置文件在哪里
    • PHP Composer是什么技术?一起看看
    • phpmyadmin如何运行
    • PHP重定向如何实现数据不丢失?
    • 详解PHP的反射使用
    • ThinkPHP6.0 数据库链式操作
    • ThinkPHP6.0 杂项
    • PHP EOF(heredoc)

    本文有爱上源码下载完入驻作者发布,如果对您版权造成侵害,可以联系本站站长管理进行维权删除,本站收到维权24小时内进行处理,谢谢您关注23ym.cn!
    本站分享大量程序员技术文章以及对编程开发的初级入门教程,包括图文讲解笔记和高清视频下载~

    重要声明:
    1.本站视频教程,软件及网站源码版权均属于原作者所有,您必须在下载后的24个小时之内,从您的电脑中删除!非法商业用途,后果自负!
    2.本站不保证所提供下载资源的安全性和完整性,仅供下载学习之用!如链接失效或资源含外站广告,请联系客服处理!给予奖励!
    3.本站所有资源来源于用户上传和网络,因此不包含技术服务请大家谅解!本站提供有偿服务!如有侵权请联系在线客服!
    4.如您手中有优质资源或教程,可以自助投稿发布,成功分享后有奖励和额外收入!
    5.如您需要正版微擎模块可联系本站客服,我们有价值30w+商业微擎应用出售微擎坑位和招收代理!
    6.400电话/软著/ICP,EDI许可证/商标特价办理中!
    爱上源码下载网 » 详解PHP的反射使用

    常见问题FAQ

    从网站下载的源码都有安装教程么?不会安装怎么办?
    本站发布的网站源码和模板资源大部分在压缩包内都有教程,如您不会安装可以联系本站在线技术进行付费安装。
    爱上源码的所有源码都是亲测能正常运行的么?
    本站目前拥有资源10w+,包含整站源码,网站模板,游戏源码,小程序源码,视频教程,破解软件等,每天也在测试更新;因时间和精力有限我们无法对资源进行一一测试,只能保证所分享资源内容无误,希望理解。
    我手中的优质资源可以在你这换钱或者VIP么?
    爱上源码支持投稿,欢迎发布您手中的优质资源进行售卖;本站VIP支持免费获取,目前邀请10人注册爱上源码即可免费获取VIP。
    爱上源码除了资源分享还有其他业务没?
    【价值30W+微擎模块出售正版商业微擎坑位及招收代理,详情咨询本站客服!】我们团队目前运营并推广几套商业化saas智能小程序系统能满足大部分小程序开发需求,并由SaaS和独立部署版商城小程序系统;另外销售400电话,各种ICP/EDI资质证书办理,软著和商标注册服务等。

    发表评论

    • 32会员总数(位)
    • 35644资源总数(个)
    • 0本周发布(个)
    • 0 今日发布(个)
    • 666稳定运行(天)

    提供最优质的资源集合

    开通VIP 源码下载