Zend_Controller进阶
在前边的例子中,我们的index.php引导文件所在的文件夹与控制器、视图等所在的文件夹是不同的,这并不符合一般网站项目的文件夹的组织习惯。从本部分开始,我们把他们放在同一个文件夹中,这是因为ZF提供了灵活的文件夹组织和配置能力。
我们新的文件夹结构如下: …/htdocs
library
Zend
test.zf.com
controllers
models
views
index.php
.htaccess
即把原来app_phpchina1.com文件夹下的子文件夹Controllers、models、views转移到文件夹phpchina1.com下,与index.php和.htaccess文件放在一起。为了不覆盖原来的代码,我们把原来的phpchina1.com文件夹拷贝一份,命名为test.zf.com。
为了测试,我们建立一个新的虚拟主机test.zf.com。这可以仿照前面的phpchina1.com来完成,即在httpd.conf中添加:<VirtualHost *:8080>
ServerAdmin any@any.com
DocumentRoot “C:/Program Files/Apache Software Foundation/Apache2.2/htdocs/test.zf.com”
ServerName test.zf.com
ErrorLog “logs/test.zf.com-error.log”
CustomLog “logs/test.zf.com-access.log” common
</VirtualHost>
重新启动Apache服务。
然后再在hosts文件中加一条
<服务器IP> phpchina1.com
以后即可以以http://test.zf.com:8080/*.*的形式访问新的网站。但现在暂时还不能像前边的例子一样运行原来的示例代码,因为还需要对index.php里的路径配置进行调整。
我们把语句PATH_SEPARATOR . ‘../App_phpchina1.com/models/’修改为:
PATH_SEPARATOR . ‘models/’
把语句”default”=>’../app_phpchina1.com/controllers’修改为:
“default”=>’controllers’
然后用浏览器打开地址http://test.zf.com:8080,就可以看到和原来一样,显示了字符串“Hello PHPChina1.com!”。也可以打开http://test.zf.com:8080/news显示“Welcome to News!”字符串。
下一步我们对Controllers进行扩展改造。比如我们有这样的需求:如果一个网站项目很大,有多个模块,Controllers文件夹下将有大量的控制器文件,我们打算通过按模块建立子文件夹对其进行分类存放,IndexController.php控制器文件存放在index文件夹下,news控制器文件存放在news文件夹下。
首先修改index.php文件,把
$fc->setControllerDirectory(array(
“default”=>’controllers’,
));
处的”default”=>’controllers’修改为
“default”=>’controllers/index’
这时http://test.zf.com:8080地址就可以打开了,显示结果与原来相同。
但要访问news文件夹下的NewsController.php控制器,必须把news添加到搜索路径,并为其命名,我们这里命名为other模块,这样我们就添加以下语句:
$fc->addControllerDirectory(’controllers/news’, ‘other’);
然后打开地址http://test.zf.com:8080/news,看不到原来的结果。
再打开地址http://test.zf.com:8080/other和http://test.zf.com:8080/other/news,也看不到原来的结果。
原来,需要修改NewsController.php中的语句
Class NewsController extends Zend_Controller_Action为
class other_NewsController extends Zend_Controller_Action
即在类名前加“other_”字符串。然后访问地址
http://test.zf.com:8080/other/news
即可得到正确的结果。
这里有一个新问题,即http://test.zf.com:8080/other/news和index默认文件夹的
OtherController控制器中的newsAction方法访问路径是相同的,如果index文件夹下同时还有文件OtherController.php,内容为:<?php
class OtherController extends Zend_Controller_Action
{
function newsAction()
{
echo “This is OtherController.php ==> newAction”;
}
}
?>
那么http://test.zf.com:8080/other/news会不会显示
This is OtherController.php ==> newAction
字符串的结果呢?不会的,因为按模块搜索访问方式优先。有了other搜素路径名,就不能再访问默认文件夹下的other控制器中的任何方法了,即使搜素路径下没有任何对应的控制器文件,也不能访问。即,即使没有news文件夹下的NewsController.php文件,http://test.zf.com:8080/other/news也不会获得index文件夹下OtherController.php中newsAction方法的结果。
这里的搜素路径和Web服务器的虚拟路径有点相似。但是因为可以用代码指定,所以更灵活一些。后面,我们根据这一点可以灵活的组织网站项目的文件夹结构。
我们对前边ZF中请求URL的格式做一些补充,原URL格式
http://host_name/controller_name/action_name/param1/ value1/param2/ value2…
可以扩展为:
http://host_name/路径/controller_name/action_name/param1/ value1/param2/ value2…
路径名需要作为“路径名_”字符串的形式加在
代码“Class controller_name”的实现代码的前边,成为“Class路径名_controller_name”的形式。明白了这点,在分析ZF的源代码的时候,就不会被路径问题所迷惑了。