2

通常这不是我会问的那种问题,因为我认为我错过了一些愚蠢的东西,但是在调试中我发现了一些非常奇怪的行为,这真的让我很难过。

描述发生(疑似)错误的设置:我有一个基本的 CRUD 控制器(或在我的情况下首选的 NERD),其中我有四个操作:

  • 新的
  • 编辑
  • 删除

读取、编辑和删除操作需要一个对象,例如一篇博文,并且/post/read/[:id]需要提供由 id 之类的路由调用的 url。我在每个操作中做的第一件事是检查 (a) id 参数是否已设置以及 (b) id 是否对应于存储在数据库中的有效实体(我使用 Doctrine 2 作为映射器)。如果 (a) 或 (b) 失败,我会立即返回并重定向到 index 操作:

 return $this->redirect()->toRoute('post/index');

我在同一个控制器中的一个单独的公共函数中进行检查,以防止代码重复。

编码:

<?php

namespace Blog\Controller;

use Zend\Mvc\Controller\AbstractActionController;
use Zend\Mvc\Controller\Plugin\FlashMessenger as FlashMessenger;

class PostController extends AbstractActionController
{

    public function editAction()
    {
        $post_id = $this->params()->fromRoute('id');
        $post    = $this->validatePostId($post_id);

        // Create/Validate forms and persist/flush

        return array(
            'post' => $post,
            'form' => $form
        );
    }

    public function deleteAction()
    {
        $post_id = $this->params()->fromRoute('id');
        $post    = $this->validatePostId($post_id);

        // Remove post from database
        $this->getBlogService()->deletePost($post);
        $this->flashMessenger()->addSuccessMessage('Post successfully deleted');
        return $this->redirect()->toRoute('post');
    }

    public function readAction()
    {
        $post_id = $this->params()->fromRoute('id');
        $post    = $this->validatePostId($post_id);

        return array(
            'post' => $post
        );
    }

    /**
     * Checks if $id from url is set and tries to find the corresponding post
     */
    public function validatePostId($post_id)
    {        
        if (!$post_id) {
            $this->flashMessenger()->addErrorMessage('Invalid post id');
            return $this->redirect()->toRoute('post');
        }

        $post = $this->getBlogService()->getPostById($post_id);

        if ($post == NULL) {
            $this->flashMessenger()->addErrorMessage('Invalid post id');
            return $this->redirect()->toRoute('post');
        }

        return $post;
    }
}

问题

这段代码完美地工作,除了readAction()即使两个条件validatePostId()触发,return $this->redirect()也不会以某种方式触发,并且应用程序呈现视图和随后的错误以及对 non-object 的无效访问$post

我已经检查了一千次,并且确实输入了检查有效的条件,$post因此我确定调用了重定向。我发现但无法解释的提示如下。如果我将读取操作更改为以下内容,只需添加一行:

public function readAction()
{
    $post_id = $this->params()->fromRoute('id');
    $post    = $this->validatePostId($post_id);

    $form    = $this->getPostForm()->bind();

    return array(
        'post' => $post
    );
}

问题消失了,应用程序巧妙地重定向。然而,这没有意义。该行只是在控制器中调用一个公共函数来实例化一个表单,据我所知,这应该没有效果。更糟的是,如果我->bind()从那条线路中删除呼叫,旧问题又回来了。所以我所能弄清楚的是,动作的区别似乎是在调用validatePostId()和返回视图之间没有任何额外的代码,这并不重要......

编辑:

更新,如果我直接使用validatePostId()函数中的代码readAction()问题也消失了:

public function readAction()
{
    $post_id = $this->params()->fromRoute('id');
    $post    = $this->validatePostId($post_id);

    if (!$post_id) {
        $this->flashMessenger()->addErrorMessage('Invalid post id');
        return $this->redirect()->toRoute('post');
    }

    $post = $this->getBlogService()->getPostById($post_id);

    if ($post == NULL) {
        $this->flashMessenger()->addErrorMessage('Invalid post id');
        return $this->redirect()->toRoute('post');
    }

    return array(
        'post' => $post
    );
}
4

1 回答 1

3

在控制器中调用的行为$this->redirect()不会自动重定向到新位置,它只是返回一个Response对象,然后您需要从控制器操作返回该对象以使请求短路。

由于您redirect()要从另一个函数返回调用结果,因此您需要首先测试该调用的结果是否是一个Response对象并在您的控制器操作中处理它......

<?php

namespace Blog\Controller;

use Zend\Mvc\Controller\AbstractActionController;
use Zend\Mvc\Controller\Plugin\FlashMessenger as FlashMessenger;

use Zend\Stdlib\ResponseInterface as Response;

class PostController extends AbstractActionController
{

    public function readAction()
    {
        $post_id = $this->params()->fromRoute('id');
        $post    = $this->validatePostId($post_id);

        // check to see if post validation returned a Response 
        if ($post instanceof Response) {
            // redirect...
            return $post;
        }

        return array(
            'post' => $post
        );
    }

    /**
     * Checks if $id from url is set and tries to find the corresponding post
     */
    public function validatePostId($post_id)
    {        
        if (!$post_id) {
            $this->flashMessenger()->addErrorMessage('Invalid post id');
            return $this->redirect()->toRoute('post');
        }

        $post = $this->getBlogService()->getPostById($post_id);

        if ($post == NULL) {
            $this->flashMessenger()->addErrorMessage('Invalid post id');
            return $this->redirect()->toRoute('post');
        }

        return $post;
    }
}
于 2013-10-27T08:20:26.980 回答