首页 | 新闻 | 交流 | 问吧 | 文档 | 手册 | 下载 | 博客

收藏此问题 发表新评论

也是Zend_Controller_Action的几个使用技巧

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的PluginsAction Helpers 好像很强大的样子(因为不懂,所以强大)。
谁有一些经验多多分享。

[ 本帖最后由 qqinxl 于 2007-8-21 00:13 编辑 ]
昵称: qqinxl  时间: 2007-08-20 23:51:00
不错的东东。谢谢。
昵称: lyw0301  时间: 2007-08-21 09:48:00
牛人,顶你一个
昵称: lichao  时间: 2007-08-21 17:57:00
现在才发现这个帖子,太牛叉了
昵称: bc1998  时间: 2008-01-13 23:14:00
写了一系列原创文章了,可惜过两天就沉了;
而且,现在对Zend Framework热情也过了,不爱写东西了。

论框架,还是看J2EE领域阿。
总觉得Zend Framework缺些什么。
比如国际化问题,觉得http://bbs.phpchina.com/viewthread.php?tid=44407的方法还是不够优雅。
昵称: qqinxl  时间: 2008-01-16 10:53:00
文章不错,欢迎来phpeye论坛:)

关于多语言应用,试试Zend_Translate。
昵称: Haohappy  时间: 2008-01-16 15:22:00
但是把MyControllerAction部署在什么位置比较合理而且应用起来比较好呢?放在控制器下直接继承是找不到文件的.除非你引用此文件.放在其他位置看起来又有点不太合道理
昵称: bc1998  时间: 2008-01-16 21:11:00