1

我没有看到一条路线中涉及多个蛞蝓的问题。

我有一个名为“教师”、“课程”和“科目”的表格。每个都有一个名为“快捷方式”的字段。

如何从如下网址中路由和提取正确的 Teacher.id、Course.id 和 Subject.id:

http://www.domain.com/bowlerae/grade-5/math

其中“bowlerae”是 Teacher.shortcut,“grade-5”是 Course.shortcut,“math”是 Subject.shortcut。

我不确定最初如何路由,如何提取每 3 个元素的 ID(我是在 AppController 中还是在我需要的每个单独的控制器中执行此操作?)以及如何构建与反向路由和分页兼容的链接。此外,“快捷方式”字段是否需要为每个表具有唯一名称,例如“teacher_slug”、“course_slug”、“subject_slug”?

现在,我正在将 3 个模型中的每一个加载到我的 routes.php 中并查询以获取每个模型的相应 ID(如果 URL 的该部分存在),然后将这些值传递给控制器​​。但正如你可以想象的那样,它花费了大量的资源。

其他问题/疑虑(针对 AD7six 和其他)

如果我想要这样的网址

http://www.domain.com/bowlerae

指向教师的主页,或类似这样的 URL

http://www.domain.com/bowlerae/grade-5

指向课程的主页,我会有如下路线

Router::connect(
    '/:teacher', 
    array('controller' => 'teachers', 'action' => 'view'),
    array(
        'teacher'   => '[a-z0-9]{1}([a-z0-9\-]{2,}[a-z0-9]{1})?',
        'pass' => array('teacher')
));

对于教师,以及以下课程

Router::connect(
    '/:teacher/:course', 
    array('controller' => 'courses', 'action' => 'view'),
    array(
        'teacher'   => '[a-z0-9]{1}([a-z0-9\-]{2,}[a-z0-9]{1})?',
        'course'   => '[a-z0-9]{1}([a-z0-9\-]{2,}[a-z0-9]{1})?',
        'pass' => array('teacher', 'course')
));

我担心的是,如果有人导航到诸如

http://www.domain.com/account其中帐户是有效的控制器但不是有效的教师

或者

http://www.domain.com/bowlerae/students学生是有效的控制者但不是有效的课程

我将如何处理这些情况?我尝试像这样使用重定向方法......

TeachersController.php

public function view($tSlug = null) {
    $this->Teacher->id = $this->Teacher->findByShortcut($tSlug);

    if (!$this->Teacher->exists()) {
                  $this->Session->setFlash('This teacher does not exist');
                  $this->redirect(array('controller' => $tSlug));
    }
            else{
            // do this
            } // end else if teacher exists
    } 

我希望它路由以下 URL

http://www.domain.com/account到帐户控制器,但将其路由到

http://www.domain.com/account/index,正确的帐户控制器和正确的索引操作,但它将“/index”附加到 URL 的末尾。

以我现在的路线方式,它正在尝试访问课程控制器,所以我向课程控制器添加了相同的重定向方法,就像这样......

CoursesController.php

public function view($tSlug = null, $cSlug = null) {
    $this->Teacher->id = $this->Teacher->findByShortcut($tSlug);
    $this->Course->id = $this->Course->findByShortcut($cSlug);


    if (!$this->Teacher->exists()) {
                  $this->Session->setFlash('This teacher does not exist');
                  $this->redirect(array('controller' => $tSlug, 'action' => $cSlug));
    }
    elseif (!$this->Course->exists()) {
                  $this->Session->setFlash('This course does not exist');
                  $this->redirect(array('controller' => $tSlug, 'action' => $cSlug));
    }
            else{
            // do this
            } // end else if everything is fine
    } 

不幸的是,我收到一条错误消息,指出重定向永远不会结束。

此外,我已经为这样的 URL 创建了路由(如果没有教师、课程或主题),但它们可能不正确。

4

1 回答 1

3

针对 db 测试所有请求以查看它们是否匹配路由是一个非常糟糕的主意。

将逻辑放入控制器中

一种简单的方法是创建路由 ala:

Router::connect('/*', array('controller' => 'x', 'action' => 'y'));

或者更具体一点:

Router::connect(
    '/:teacher/:course/:subject', 
    array('controller' => 'x', 'action' => 'y'),
    array('pass' => array('teacher', 'course', 'subject')
);

并确保首先声明您使用的所有其他路线。然后简单地创建一个控制器动作,如下所示:

XController extends AppController {

    public $uses = array('Teacher');

    public function y($tSlug, $cSlug, $sSlug) {
        $teacher = $this->Teacher->findBySlug($tSlug);
        ...
    }

}

限制路由元素

使用路由验证,您可以限制与给定路由元素匹配的内容,例如,如果只有 5 位教师,您可以通过确保以他们的 slug 开头来防止任何其他 url 匹配:

Router::connect(
    '/:teacher/:course/:subject', 
    array('controller' => 'x', 'action' => 'y'),
    array(
        'teacher' => 'rita|sue|bob|larry|mo',
        'pass' => array('teacher', 'course', 'subject')
);

如果您想从数据库中读取教师的 slug -缓存数据库查询,请不要在所有请求的路由文件中读取数据库。

在您的路线中放置一些固定的东西

使用没有固定字符串的路由意味着任何带有 3 个路径段的 url 都将匹配该路由 - 在路由中放置一些固定的东西可能不太容易出错,这样您就可以确定该请求实际上是针对老师的, 课程, 科目组合

IE

Router::connect('/anything/*', array('controller' => 'x', 'action' => 'y'));

这样就不需要任何“此 url 是否匹配此路由?” 逻辑 - 是这样,或者不是。

于 2013-03-25T20:43:21.450 回答