+ -
当前位置:首页 → 问答吧 → 也是Zend_Controller_Action的几个使用技巧

也是Zend_Controller_Action的几个使用技巧

时间:2007-08-21

来源:互联网

xgwork 的帖子里提到了Zend_Controller_Action的一个技巧,就是继承。

也贴一些实在的代码,大家讨论一下。也许有更技巧的方法,希望有经验的多多提意见。
1、验证问题。
2、编码转换
3、相同的Controller指向不同的View
4、_redirect函数重构
5、baseUrl的问题

我们都知道,在Zend_Controller_Action里,任何一个请求,都是controller/action构成;而任何一个请求的执行顺序都是init() preDispatch() xxAction() postDispatch()
利用这个,我们可以做很多事情。

1、验证问题。
其实,这个实例大家可以在Getting Started with Zend_Auth by Rob Allen (English)  里找到。当然,英文资料里只是简单实现,很多功能我们可以再扩展。
一个复杂的验证问题是不同页面可能要求验证对象不同。
比如A页面不需要验证。B页面必须登陆用户才能显示;而C必须登陆管理员才能显示。

通过Zend_Controller_Action 继承重构,就可以解决这个问题。

很明显,不需要验证的A页面,我们可以直接直接继承Zend_Controller_Action ,或者写一个MyControllerAction 继承于Zend_Controller_Action,但不增加验证代码。
复制PHP内容到剪贴板
PHP代码:
abstract class MyControllerAction extends  Zend_Controller_Action {
}

而B页面和C页面,我们可以通过上面说的,每一个C/A 必须经过init()和preDispatch()原则,进行验证。
复制PHP内容到剪贴板
PHP代码:
abstract class MyControllerAction_User extends MyControllerAction {
//截取部分代码
public function init(){
  parent::init();
  //通过一个模块取得登陆数据
  $authMoudle = Moudle_Auth::getInstance();
  $this->_loginUser = $authMoudle->isLogin(Object_User::User );
  $this->view->loginUser = $this->_loginUser;
}
public function preDispatch(){
  //这里判断是否存在该用户,如果不存在跳出
  if (empty($this->_loginUser)) {   
     if(Mobile::isMobile()){
      $this->_redirect('/auth/ageover');
     }else{
      $this->_redirect('/auth/login');
     }
  }
}
}

通过这个方法,还可以控制那些Controller需要验证,那些不需要验证。

还可以写一个MyControllerAction_Admin用来验证其他类用户,控制C页面的显示。
复制PHP内容到剪贴板
PHP代码:
//截取部分代码
abstract class MyControllerAction_Admin extends  MyControllerAction {
public function init(){
  parent::init();
  $authMoudle = Moudle_Auth::getInstance();
  //这里就变成验证管理员身份
  $this->_loginAdmin = $authMoudle->isLogin(Object_User::Admin );
  $this->view->loginAdmin = $this->_loginAdmin;
}
public function preDispatch(){
  if (empty($this->_loginAdmin)) {   
       $this->_redirect('/admin/auth/login');
    }
}
}

这样我们的Controller文件可能这样写:
复制PHP内容到剪贴板
PHP代码:
class AController extends MyControllerAction{
//不需要验证
}
class BController extends MyControllerAction_User{
//验证用户
}
class CdviceController extends MyControllerAction_Admin{
//验证管理员
}

当然,我们再利用Zend_Controller_Front,可以构造出更丰富的逻辑层次
复制PHP内容到剪贴板
PHP代码:
$frontController->setControllerDirectory(array(
'default' => './application/controllers',
'admin' => './application/admin/controllers',
));

2、编码转换
很多项目数据库保存数据使用UTF-8,而页面显示使用另外一个编码。
我的项目就存在这个问题,页面是SJIS码。数据库等交换保存的全部是UTF-8码。

一个很简单的实现办法就是重构Zend_Controller_Action中的render()方法。注意不是Zend_View里的render()。
大家应该在Controller中几乎每个Action最后一句都是$this->render()。我们就是通过控制这个视图显示,达到编码转换。
复制PHP内容到剪贴板
PHP代码:
//第四个参数是为ajax设置的。我利用的prototype.js,取得数据自动转为当前页面码,所以不需要提前转换
public function render($action = null, $name = null, $noController = false,$ajax = false)
{
  $view   = $this->initView();
  $script = $this->getViewScript($action, $noController);
  if($ajax){
   $this->getResponse()->appendBody($view->render($script),$name);
  }else{
   $this->getResponse()->appendBody(mb_convert_encoding($view->render($script), "SJIS",'UTF-8'),$name);
  }
}

既然取数据显示到页面需要Utf-8转到SJIS,那么页面取得$_GET或者$_POST也是需要SJIS转到UTF-8才能保存到数据库,或者其他操作的。
因此需要再增加以下两个函数用来获取用户提交的$_GET或者$_POST
复制PHP内容到剪贴板
PHP代码:
public function getUtfParam($key){
  try{
   $_paras = $this->getRequest()->getParam($key);
   if(!empty($_paras) and is_string($_paras)){
    $_paras = mb_convert_encoding($_paras,'UTF-8', "SJIS,JIS,sjis-win");
   }
   return $_paras;
  }catch (Exception $e){
   throw $e;
  }
}
public function getUtfPost($key){
  try{
   if($this->getRequest()->isPost()){
    $filter = new Zend_Filter_StripTags();
    $value= $filter->filter($this->getRequest()->getPost($key));
    $value = trim($value);
    if(!empty($value) and is_string($value)){
     $value = mb_convert_encoding($value,'UTF-8', "SJIS,JIS,sjis-win");
    }
    return $value;
   }else{
    return null;
   }
  }catch (Exception $e){
   throw $e;
  }
}

3、相同的Controller指向不同的View
同时开发手机Web和Pc Web的时候,我们都知道手机界面和Pc界面差异很大。
一般情况下几乎完全分离。
但是,实际上我们的服务器端数据确实几乎一样的。
比如,用户请求一个Url http://localhost/user/show/id/111 显示id=111的用户信息
服务器端,也就是我们的controller取得数据一样,但是客户端显示的html确分Pc版和mobile版

重构重构Zend_Controller_Action中的initView()方法可以提供一个很好的解决方案。
复制PHP内容到剪贴板
PHP代码:
public function initView()
{
  if (!$this->getInvokeArg('noViewRenderer') && $this->_helper->hasHelper('viewRenderer')) {
   return $this->view;
  }
  require_once 'Zend/View/Interface.php';
  if (isset($this->view) && ($this->view instanceof Zend_View_Interface)) {
   return $this->view;
  }

  $request = $this->getRequest();
  $module  = $request->getModuleName();
  $dirs    = $this->getFrontController()->getControllerDirectory();
  if (empty($module) || !isset($dirs[$module])) {
   $module = 'default';
  }
  $baseDir = dirname($dirs[$module]) . DIRECTORY_SEPARATOR . 'views';
  if (!file_exists($baseDir) || !is_dir($baseDir)) {
   throw new Zend_Controller_Exception('Missing base view directory ("' . $baseDir . '")');
  }
  //在这里,我们重新写了[b]$baseDir[/b] ,指向不同目录
if(Mobile::isMobile()){
   $baseDir .= DIRECTORY_SEPARATOR .'mobile';
  }else{
   $baseDir .= DIRECTORY_SEPARATOR .'pc';
  }
  require_once 'Zend/View.php';
  $this->view = new Zend_View(array('basePath' => $baseDir));
  return $this->view;
}

这时候目录结构应该是这样的:
复制PHP内容到剪贴板
PHP代码:
/application
    /controllers
    /models
    /views
       /mobile
            /filters
            /helpers
            /scripts
       /pc
            /filters
            /helpers
            /scripts

在views结构下,可以分离为/mobile和/pc两个目录;而两个目录的数据受同一个controller控制。

其实,这个应用不一定手机版Pc版。我们可以在这里控制不同的View版本,比如:
复制PHP内容到剪贴板
PHP代码:
if(Version::isVersion('1.0')){
   $baseDir .= DIRECTORY_SEPARATOR .'version1';
}elseif(Version::isVersion('2.0')){
  $baseDir .= DIRECTORY_SEPARATOR .'version2';
}else{
   $baseDir .= DIRECTORY_SEPARATOR .'version-test';
  }

4、_redirect函数重构
重构是因为$this->_redirect('user/show');竟然出现错误。(1.0Rc3版本测试结果)
正确的写法是$this->_redirect('/user/show');
因此重写_redirect函数。同时增加session id,支持不支持cookie的浏览器。
复制PHP内容到剪贴板
PHP代码:
protected function _redirect($url, array $options = array()){
  if(substr($url,0,strlen('/')) === '/'){
   $url = trim($url);
  }else{
   $url = '/'.trim($url);
  }
  $url = $url.'?'.SID;
  parent::_redirect($url,$options);
}

5、baseUrl的问题
这个问题也许有更好的解决办法。
在Rob Allen 的教材里,其实有一个bug。如果url写成http://localhost/zf-tutorials/index.php 的话,css不能正确取得。
其实这个baseUrl 可以重写的。
复制PHP内容到剪贴板
PHP代码:
protected function _init(){
  $url = $this->_request->getBaseUrl();
  $this->_baseUrl = (basename($url)=='index.php')?dirname($url):$url;
  $this->view->baseUrl = $this->_baseUrl;
}

项目缘故,以上重构是zf版本0.9时候开始写的。经测试兼容1.0。

最近看了一些Zend_Controller_Action的Plugins 和Action Helpers 好像很强大的样子(因为不懂,所以强大)。
谁有一些经验多多分享。

[ 本帖最后由 qqinxl 于 2007-8-21 00:13 编辑 ]

作者: qqinxl   发布时间: 2007-08-20

不错的东东。谢谢。

作者: lyw0301   发布时间: 2007-08-21

牛人,顶你一个

作者: lichao   发布时间: 2007-08-21

现在才发现这个帖子,太牛叉了

作者: bc1998   发布时间: 2008-01-13

写了一系列原创文章了,可惜过两天就沉了;
而且,现在对Zend Framework热情也过了,不爱写东西了。

论框架,还是看J2EE领域阿。
总觉得Zend Framework缺些什么。
比如国际化问题,觉得http://bbs.phpchina.com/viewthread.php?tid=44407的方法还是不够优雅。

作者: qqinxl   发布时间: 2008-01-16

文章不错,欢迎来phpeye论坛:)

关于多语言应用,试试Zend_Translate。

作者: Haohappy   发布时间: 2008-01-16

但是把MyControllerAction部署在什么位置比较合理而且应用起来比较好呢?放在控制器下直接继承是找不到文件的.除非你引用此文件.放在其他位置看起来又有点不太合道理

作者: bc1998   发布时间: 2008-01-16

热门下载

更多