设为首页 收藏本站
查看: 1231|回复: 0

[经验分享] 明晰php Zend Framework中的render

[复制链接]
累计签到:1 天
连续签到:1 天
发表于 2015-8-23 11:13:01 | 显示全部楼层 |阅读模式
搞asp.net mvc时对php mvc发生兴趣 看了看Zend Framework(下简称ZF)
可以对比理解mvc
对一个比较核心的函数render很是迷惑了一阵 记录下明晰过程
通常在我们利用ZF实现php的mvc时,最关键的地方当然是Controller类的各种action方法,在action方法中,我们确定及输出内容. 在类 abstract class Zend_Controller_Action 中的dispatch方法你可以发现这一行 $this->$action();
那么如何确定及输出内容呢,就是进行render了,不过这个render却是有好几个的,下面列出这几个情形

1 DSC0000.gif <?php
2class IndexController extends Zend_Controller_Action
3{
4    public function contactAction()
5    {
6        //$this->render("index");
7        //$this->render();
8        //$this->renderScript("sidebar.phtml");
9
10
11        //$this->_helper->viewRenderer("sidebar");
12
13        //$this->view->render("sidebar.phtml");
14        //$this->view("sidebar");
15
16    }
17}
18?>
总结下来,似乎就是这三中render了(欢迎补充)
1.自身render
先看第一种
  //$this->render("index");
  //$this->render();
  //$this->renderScript("sidebar.phtml");
这是直接使用Zend_Controller_Action类的render方法
第一句是render了另一个action所对应的视图(看清了 是render那个action对应的视图 而不是执行那个action!)
第二句式render本action对应的视图,这个有什么意义呢(因为很多情形你看不到这个写法的),这个下面再说.
第三句是render特定的视图文件,这里你可能认为前两个方法实际是调用了这个renderScript,其实不是如此.
下面就阐述一下.顺便解释第二句的原因.
Zend_Controller_Action类的render方法中其实是有两个分支的 如下render函数代码

1    public function render($action = null, $name = null, $noController = false)
2    {
3        if (!$this->getInvokeArg('noViewRenderer') && $this->_helper->hasHelper('viewRenderer')) {
4            return $this->_helper->viewRenderer->render($action, $name, $noController);
5        }
6
7        $view   = $this->initView();
8        $script = $this->getViewScript($action, $noController);
9
10        $this->getResponse()->appendBody(
11            $view->render($script),
12            $name
13        );
14    }可以看到一种情形是利用(代理)了视图助手类(viewRenderer)的render方法
另一种是禁用助手时的情形 就得亲自上阵了,这也就是render()出现的原因,你禁用了视图助手后要输出本action对应视图内容可以使用render()来完成
2.通过视图助手viewRenderer
上面说起了视图助手,那我们来看action中的第二个片段,正是借助视图助手来进行
    //$this->_helper->viewRenderer("sidebar");
实际上这里这句话并不是render内容,而是指定了要render哪个视图,参考Zend_Controller_Action_Helper_ViewRenderer类的这个函数

1    public function direct($action = null, $name = null, $noController = null)
2    {
3        $this->setRender($action, $name, $noController);
4    }那么输出呢 是怎么输出的?
可以在$this->_helper->viewRenderer("sidebar"); 后直接调用$this->render();即可.
但是实际上你完全不用调用,只写那一句就行.
你不写render的时候,视图助手会来替你完成.在Zend_Controller_Action类中的dispatch方法中有这么一句
$this->_helper->notifyPostDispatch();
_helper是什么? 是一个Zend_Controller_Action_HelperBroker类 ,其中有这个方法

1    public function notifyPostDispatch()
2    {
3        foreach (self::getStack() as $helper) {
4            $helper->postDispatch();
5        }
6    }
可以看到调用了其中各个助手的postDispatch();
而viewRenderer正是其中的一个助手,其postDispatch方法如下

1    public function postDispatch()
2    {
3        if ($this->_shouldRender()) {
4            $this->render();
5        }
6    }正是在这里视图助手帮你进行了render,如果你自己render了,聪明的视图助手会知晓的,可以查看下在_shouldRender()中的这个 $this->getRequest()->isDispatched(),及Zend_Controller_Front 类中dispatch方法的这句话:$this->_request->setDispatched(true);
3.终极render 关于Zend_View->render()
好了现在我们来看看Zend_View的render().
在上面的两个中我们都说到了render(),比如action的render和视图助手的render
那么你该问个问题:就这样了?后面呢?
后面的才是关键的.
在action的render中,你可能注意到这句话了
10        $this->getResponse()->appendBody(
11            $view->render($script),
12            $name
13        );
而我们再看看viewRenderer的render(),viewRenderer的render方法其实是调用了renderScript方法,代码如下

1    public function renderScript($script, $name = null)
2    {
3        if (null === $name) {
4            $name = $this->getResponseSegment();
5        }
6
7        $this->getResponse()->appendBody(
8            $this->view->render($script),
9            $name
10        );
11
12        $this->setNoRender();
13    }可以看到这里跟action的render有点类似,也有同样的那句话.
就是说action的render和viewRenderer的render其实都是调用Zend_View的render,拿到内容而后置放到response中
Zend_View的render:

1    public function render($name)
2    {
3        // find the script file name using the parent private method
4        $this->_file = $this->_script($name);
5        unset($name); // remove $name from local scope
6
7        ob_start();
8        $this->_run($this->_file);
9
10        return $this->_filter(ob_get_clean()); // filter output
11    }
至于run:

1    protected function _run()
2    {
3        if ($this->_useViewStream && $this->useStreamWrapper()) {
4            include 'zend.view://' . func_get_arg(0);
5        } else {
6            include func_get_arg(0);
7        }
8    }
那么你就明白了最开始代码中的第13行
13        //$this->view->render("sidebar.phtml");
其实是个幌子,哈.这句话只是得到了内容,但是呢 没做处理!
所以我们应该这样
13        echo $this->view->render("sidebar.phtml");
再然后呢?参看Zend_Controller_Front类dispatch
$this->_response->sendResponse();
及Zend_Controller_Response_Abstract类

1    public function outputBody()
2    {        
3        foreach ($this->_body as $content) {            
4            echo $content;
5        }
6    }
而至于第14行
14        //$this->view("sidebar");
貌似合理,瞪一眼就知道了:这句话地地道道的错误
action 中没有这个方法,__call中也没有相应处理,不象_helper->viewRenderer("sidebar");在 _helper针对该情况在__call中有相应处理

1    public function __call($method, $args)
2    {
3        $helper = $this->getHelper($method);
4        if (!method_exists($helper, 'direct')) {
5            require_once 'Zend/Controller/Action/Exception.php';
6            throw new Zend_Controller_Action_Exception('Helper "' . $method . '" does not support overloading via direct()');
7        }
8        return call_user_func_array(array($helper, 'direct'), $args);
9    }没有viewRenderer这个方法,于是去寻找名为viewRenderer并且有direct方法的助手,找到了即执行这个direct方法(上面第二部分贴过代码了)
至于viewRenderer这个助手存放时,要注意到他的名字是死的 就是"viewRenderer",具体看Zend_Controller_Action_Helper_Abstract类的getName方法

1    public function getName()
2    {
3        $full_class_name = get_class($this);
4
5        if (strpos($full_class_name, '_') !== false) {
6            $helper_name = strrchr($full_class_name, '_');
7            return ltrim($helper_name, '_');
8        } else {
9            return $full_class_name;
10        }
11    }之所以提到这点是因为在Zend_Controller_Action_Helper_ViewRenderer注释中你能看到这句话
// In your action controller methods:
$viewHelper = $this->_helper->getHelper('view');
而实际上你复制这句话到your action controller methods中去 只会出错
Exception information:
  Message: Action Helper by name View not found

  

个人感觉php Zend Framework还是很不错,虽然一直没搞好调试器
但是非常满意于可以随处置放var_dump

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-102915-1-1.html 上篇帖子: 关于PHP位运算的简单权限设计 下篇帖子: PHP内核探索之变量(4)- 数组操作
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表