2008-10-02 47 views
12

在Kohana中/笨,我可以有一个URL形式如下:使用URL段在Zend框架的行动方法的参数

http://www.name.tld/controller_name/method_name/parameter_1/parameter_2/parameter_3 ... 

,然后阅读在我的控制器参数如下:

class MyController 
{ 
    public function method_name($param_A, $param_B, $param_C ...) 
    { 
     // ... code 
    } 
} 

你如何在Zend框架中实现这一点?

回答

6

更新(2016年4月13日): 在下面移动,并已固定我的答案的链接。但是,以防万一再次消失了 - 这里有提供一些深入的信息,这种技术的几个备选方案,以及使用原来的文章作为参考材料:


@Andrew Taylor的回复是正确的Zend Framework处理URL参数的方式。但是,如果您想在控制器的操作中使用URL参数(如您的示例中所示) - 请查看this tutorial on Zend DevZone

11

在Zend_Controller_Router类请看:

http://framework.zend.com/manual/en/zend.controller.router.html

这将允许你定义一个Zend_Controller_Router_Route中映射到您的网址,你需要的方式。

具有用于索引控制器的索引行为4个静则params的一个例子是:

$router = new Zend_Controller_Router_Rewrite(); 

$router->addRoute(
    'index', 
    new Zend_Controller_Router_Route('index/index/:param1/:param2/:param3/:param4', array('controller' => 'index', 'action' => 'index')) 
); 

$frontController->setRouter($router); 

这是添加到您的引导已经定义了你的前端控制器之后。

一旦你的行动,你就可以使用:

$this->_request->getParam('param1'); 

内,您的操作方法来访问值。

安德鲁

+0

尼斯,该文档是不是在考虑那么清楚。 – 2010-04-13 07:04:58

1

原本张贴在这里http://cslai.coolsilon.com/2009/03/28/extending-zend-framework/

我目前的解决方案如下:

abstract class Coolsilon_Controller_Base 
    extends Zend_Controller_Action { 

    public function dispatch($actionName) { 
     $parameters = array(); 

     foreach($this->_parametersMeta($actionName) as $paramMeta) { 
      $parameters = array_merge( 
       $parameters, 
       $this->_parameter($paramMeta, $this->_getAllParams()) 
      ); 
     } 

     call_user_func_array(array(&$this, $actionName), $parameters); 
    } 

    private function _actionReference($className, $actionName) { 
     return new ReflectionMethod( 
      $className, $actionName 
     ); 
    } 

    private function _classReference() { 
     return new ReflectionObject($this); 
    } 

    private function _constructParameter($paramMeta, $parameters) { 
     return array_key_exists($paramMeta->getName(), $parameters) ? 
      array($paramMeta->getName() => $parameters[$paramMeta->getName()]) : 
      array($paramMeta->getName() => $paramMeta->getDefaultValue()); 
    } 

    private function _parameter($paramMeta, $parameters) { 
     return $this->_parameterIsValid($paramMeta, $parameters) ? 
      $this->_constructParameter($paramMeta, $parameters) : 
      $this->_throwParameterNotFoundException($paramMeta, $parameters); 
    } 

    private function _parameterIsValid($paramMeta, $parameters) { 
     return $paramMeta->isOptional() === FALSE 
      && empty($parameters[$paramMeta->getName()]) === FALSE; 
    } 

    private function _parametersMeta($actionName) { 
     return $this->_actionReference( 
       $this->_classReference()->getName(), 
       $actionName 
      ) 
      ->getParameters(); 
    } 

    private function _throwParameterNotFoundException($paramMeta, $parameters) { 
     throw new Exception(”Parameter: {$paramMeta->getName()} Cannot be empty”); 
    } 
} 
3

对于允许更复杂的配置更简单的方法,尝试this post。总结:

创建application/configs/routes.ini

routes.popular.route = popular/:type/:page/:sortOrder 
routes.popular.defaults.controller = popular 
routes.popular.defaults.action = index 
routes.popular.defaults.type = images 
routes.popular.defaults.sortOrder = alltime 
routes.popular.defaults.page = 1 
routes.popular.reqs.type = \w+ 
routes.popular.reqs.page = \d+ 
routes.popular.reqs.sortOrder = \w+ 

添加到bootstrap.php

// create $frontController if not already initialised 
$frontController = Zend_Controller_Front::getInstance(); 

$config = new Zend_Config_Ini(APPLICATION_PATH . ‘/config/routes.ini’); 
$router = $frontController->getRouter(); 
$router->addConfig($config,‘routes’); 
+0

但我必须为每个控制器配置:D – Jeffrey04 2010-01-05 09:12:01

+0

这经常是有用的,我倾向于找到控制器和路由组在一起,即/ page/get/1和/ user/andy/confirm/email-confirmation-token等,虽然对于更大的应用程序,它可能会变得笨拙 – Andy 2010-01-05 11:23:43

4

我有我的控制器类扩展Zend_Controller_Action并进行了如下修改:

dispatch($action)方法代替

$this->$action();

call_user_func_array(array($this,$action), $this->getUrlParametersByPosition());

而在我的行动增加了以下方法

/** 
* Returns array of url parts after controller and action 
*/ 
protected function getUrlParametersByPosition() 
{ 
    $request = $this->getRequest(); 
    $path = $request->getPathInfo(); 
    $path = explode('/', trim($path, '/')); 
    if(@$path[0]== $request->getControllerName()) 
    { 
     unset($path[0]); 
    } 
    if(@$path[1] == $request->getActionName()) 
    { 
     unset($path[1]); 
    } 
    return $path; 
} 

现在的URL像/mycontroller/myaction/123/321

我会得到以下所有控制器的PARAMS和行动

public function editAction($param1 = null, $param2 = null) 
{ 
    // $param1 = 123 
    // $param2 = 321 
} 

URL中的其他参数不会导致任何错误,因为您可以将更多参数发送给随后定义的方法。您可以通过func_get_args() 获得所有这些信息,而且您仍可以通常的方式使用getParam()。 您的网址可能不包含使用默认网址的操作名称。

其实我的网址不包含参数名称。只有他们的价值。 (正如它在问题中一样) 而且您必须定义路径以指定URL中的参数位置,以遵循框架的概念并能够使用Zend方法构建URL。 但是如果你总是知道你的参数在URL中的位置,你可以很容易地得到它。

这并不像使用反射方法那么复杂,但我想提供更少的开销。

调度方法现在看起来是这样的:

/** 
* Dispatch the requested action 
* 
* @param string $action Method name of action 
* @return void 
*/ 
public function dispatch($action) 
{ 
    // Notify helpers of action preDispatch state 
    $this->_helper->notifyPreDispatch(); 

    $this->preDispatch(); 
    if ($this->getRequest()->isDispatched()) { 
     if (null === $this->_classMethods) { 
      $this->_classMethods = get_class_methods($this); 
     } 

     // preDispatch() didn't change the action, so we can continue 
     if ($this->getInvokeArg('useCaseSensitiveActions') || in_array($action, $this->_classMethods)) { 
      if ($this->getInvokeArg('useCaseSensitiveActions')) { 
       trigger_error('Using case sensitive actions without word separators is deprecated; please do not rely on this "feature"'); 
      } 
      //$this->$action(); 
      call_user_func_array(array($this,$action), $this->getUrlParametersByPosition()); 
     } else { 
      $this->__call($action, array()); 
     } 
     $this->postDispatch(); 
    } 

    // whats actually important here is that this action controller is 
    // shutting down, regardless of dispatching; notify the helpers of this 
    // state 
    $this->_helper->notifyPostDispatch(); 
}