Yii2创建控制器(createController)方法详解

5年以前  |  阅读数:650 次  |  编程语言:PHP 

本文实例讲述了Yii2创建控制器(createController)方法。分享给大家供大家参考,具体如下:

yii中创建控制器的是在application中的request通过UrlManager解析得出路由信息的,然后再由yii\base\Module中的


    public function runAction($route, $params = [])

方法来创建控制器,最后由控制器再执行相应的动作。

首先得明确,Yii中的路由分三种情况:

第一种是带有模块的(module id/controller id/action id),

第二种是带有命名空间(子目录)的(sub dir)/controller id/action id)

第三种是只有控制器和动作的(controller id/action id)

这三个有优先顺序,所以在创建控制器的时候,也是先查看是否是模块类型的路由,如果是,则获取这个模块,再由这个模块来创建控制器

接着再判断是否是第二种带有命名空间的。


    public function createController($route)
    {
      //如果路由为空,则使用默认的路由
      if ($route === '') {
        $route = $this->defaultRoute;
      }
      // double slashes or leading/ending slashes may cause substr problem
      //去掉首尾的反斜杠("/"),如果路由中包含有"//",则返回false创建失败。
      $route = trim($route, '/');
      if (strpos($route, '//') !== false) {
        return false;
      }
      /*
       * 路由分三种情况,
       * 一种是带模块id的(module id/controller id/action id),
       * 一种是有命名空间(子目录)的(sub dir)/controller id/action id)
       * 一种是只有控制器和动作的(controller id/action id)
       * 所以在这里要根据第一个"/"分隔成两部分,$id和$route信息,
       */
      if (strpos($route, '/') !== false) {
        list ($id, $route) = explode('/', $route, 2);
      } else {
        $id = $route;
        $route = '';
      }
      // module and controller map take precedence
      /*
       * 查看这个id是否是模块,如果是模块,则再用这个模块来创建控制器。
       * 所以,在如果一个控制器的名称和模块名称重复的话会优先创建模块里面的控制器。
       *
       * 如果有url: http://www.yii2.com/index.php?r=test/index
       * 本来是打算访问application中的控制器里面的test控制器,执行index动作的。
       *
       * 然而如果有个模块的名字为test,里面有个IndexController
       *
       * 根据上面会生成$id=test,$route=index
       *
       * 由于在下面查找存在这个模块,所以会执行这个test模块下面的index控制器,
       * 而不会执行application里面的test控制器的index动作
       */
      $module = $this->getModule($id);
      if ($module !== null) {
        return $module->createController($route);
      }
      //如果在controllerMap数组中指定了控制器映射,会优先根据这个里面的映射来创建控制器
      if (isset($this->controllerMap[$id])) {
        $controller = Yii::createObject($this->controllerMap[$id], [$id, $this]);
        return [$controller, $route];
      }
      /*
       * 如果这个时候$route中还有"/",也就是说原来的路由为home/index/aa
       * $id:home(不是模块)
       * $route:index/aa
       * 由于经过上面得知home不为模块,所以这个为命名空间(子目录),
       *
       * 再经过下面处理后为
       * $id:home/index 命名空间(子目录)home下面的index控制器
       * $route:aaa
       *
       */
      if (($pos = strrpos($route, '/')) !== false) {
        $id .= '/' . substr($route, 0, $pos);
        $route = substr($route, $pos + 1);
      }
        /*
         * $id:home/index
         * $route:aaa
         */
      $controller = $this->createControllerByID($id);
      if ($controller === null && $route !== '') {
          //如果创建失败,再加上route作为id再次创建
        $controller = $this->createControllerByID($id . '/' . $route);
        $route = '';
      }
      return $controller === null ? false : [$controller, $route];
    }

在这个函数中$id就有两种情况,一种是前面带有命名空间的,一种是直接就一个控制器ID的。


    public function createControllerByID($id)
    {
        if (!preg_match('%^[a-z0-9\\-_/]+$%', $id)) {
          return null;
        }
        /*
         * 如果$id中有"/",则前面的为目录,后面的为类
         *
         */
        $pos = strrpos($id, '/');
        if ($pos === false) {
          $prefix = '';
          $className = $id;
        } else {
          $prefix = substr($id, 0, $pos + 1);
          $className = substr($id, $pos + 1);
        }
        //生成控制器的类IndexController
        $className = str_replace(' ', '', ucwords(str_replace('-', ' ', $className))) . 'Controller';
        //如果有前缀(也就是有目录、命名空间),则在类前面加上命名空间
        $className = ltrim($this->controllerNamespace . '\\' . str_replace('/', '\\', $prefix) . $className, '\\');
        //如果类不存在,或者类名称包含"-",则出错,
        if (strpos($className, '-') !== false || !class_exists($className)) {
          return null;
        }
        //下面就是创建类了
        if (is_subclass_of($className, 'yii\base\Controller')) {
          return new $className($id, $this);
        } elseif (YII_DEBUG) {
          throw new InvalidConfigException("Controller class must extend from \\yii\\base\\Controller.");
        } else {
          return null;
        }
    }

这个过程就结束了,然后再由创建出来的控制器执行它里面的动作


    public function runAction($route, $params = [])
    {
      $parts = $this->createController($route);
      if (is_array($parts)) {
        /** @var Controller $controller */
        list($controller, $actionID) = $parts;
        $oldController = Yii::$app->controller;
        Yii::$app->controller = $controller;
        //控制器执行相应的动作
        $result = $controller->runAction($actionID, $params);
        Yii::$app->controller = $oldController;
        return $result;
      } else {
        $id = $this->getUniqueId();
        throw new InvalidRouteException('Unable to resolve the request "' . ($id === '' ? $route : $id . '/' . $route) . '".');
      }
    }

更多关于Yii相关内容感兴趣的读者可查看本站专题:《Yii框架入门及常用技巧总结》、《php优秀开发框架总结》、《smarty模板入门基础教程》、《php面向对象程序设计入门教程》、《php字符串(string)用法总结》、《php+mysql数据库操作入门教程》及《php常见数据库操作技巧汇总

希望本文所述对大家基于Yii框架的PHP程序设计有所帮助。

 相关文章:
PHP分页显示制作详细讲解
SSH 登录失败:Host key verification failed
获取IMSI
将二进制数据转为16进制以便显示
获取IMEI
文件下载
贪吃蛇
双位运算符
PHP自定义函数获取搜索引擎来源关键字的方法
Java生成UUID
发送邮件
年的日历图
提取后缀名
在Zeus Web Server中安装PHP语言支持
让你成为最历害的git提交人
Yii2汉字转拼音类的实例代码
再谈PHP中单双引号的区别详解
指定应用ID以获取对应的应用名称
Python 2与Python 3版本和编码的对比
php封装的page分页类完整实例