Symfoy2 HttpKernel事件驱动,symfoy2httpkernel
HttpKernel:事件驱动
Symfony2
框架层和应用层的工作都是在 HttpKernel::handle()
方法中完成,HttpKernel::handle()
的内部的实现其实是通过调度事件(HttpKernel
内的事件监听器)来完成的,相当于把所有组件都整合成完整的应用。
使用 HttpKernel
很简单,只需要创建一个 EventDispatcher(事件分发器) 和 controller resolver(Controller解析器),可以实现更多的事件监听器丰富应用的功能:
kernel.request Event
实现kernel.request
事件目的是为了添加更多信息到Request对象,或者得到返回的Response对象(例如:从缓存中获取又或者security层拒绝访问)
kernel.request
事件是HttpKernel::handle()
调度的第一个事件,那么监听该事件的多个监听器就会被执行。
事件的监听器多种多样,它们的行为各不相同,例如security监听器判断用户没有足够的权限的时候一个RedirectResponse对象,如果当前直接返回Response对象,那么就会直接执行 kernel.response 事件:
事件的目的要么就是直接创建和返回Response对象,要么就是添加更多的信息到Request对象。
<span>RouterListener是Symfony框架中实现</span>kernel.request<span>事件的最重要监听器,</span>RouterListener在路由层中执行,返回一个包含符合当前请求的路由信息的数组,例如路由匹配模式里面的_controller和请求的参数({name})。这些信息都会存放在Request里的attributes数组里面,目前只是会添加路由信息到Request对象中还没有做其它的动作,但是解析Controller的时候会被用到。
2) Resolve the Controller
假设实现kernel.request
事件的时候没有创建和返回Response对象,那么下一步就是确定、解析controller和controller需要的参数。controller部分是应用层的最后一个堡垒,负责创建和返回包含一个特定页面的Response对象。如何确定被请求的controller完全取决于应用程序,这个工作有controller解析器来完成——一个是实现ControllerResolverInterface的类,同时也是HttpKernel构造函数的一个参数。
首先调用controller resolver 的 getController()方法,并向该方法传入Request对象,controller resolver根据Request包含的信息确定并返回controller。
第二个方法,getArguments()会在kernel.controller事件被调度的时候执行。
<span><em>解析Controller</em></span>
<span><em>Symfony框架使用内置的ControllerResolver(实质上是使用了一个有额外的功能的子类),该解析器利用了RouterListener保存到Request对象的attributes属性里信息来确定controller。</em></span>
<span><em> </em></span>
<span><em>getController</em></span>
<em>ControllerResolver </em><em>在Request对象的attributes数组中</em><em>查找 _controller 键(这些信息实际上是由RouterListener存放进Request对象中的):</em>
<em> </em>
<em>a) 如果</em><em><span><span>_controller 键对应</span></span></em><em><span>AcmeDemoBundle:Default:index 这个格式的值,那么该值就包含了类名和方法名,可以被Symfony框架解析成为,例如:</span></em><em><span>Acme\DemoBundle\Controller\DefaultController::indexAction,这个转换是由Symfony框架的特定的</span></em><em>ControllerResolver的子类完成的。</em>
<em> </em>
<em>b) 你的controller类会被实例化,而且该controller类必须包含一个无参的构造函数。</em>
<em> </em>
<span><em>c) 如果你的controller还实现了ContainerAwareInterface,那么setContainer方法就会被调用,container就会被注入到controller中,</em></span><em><span>这个实现也是由Symfony框架的特定的</span></em><em>ControllerResolver的子类完成的。</em>
<em> </em>
<em><span>上面也有一些其他变化过程,例如你把你的controller配置成为service。</span></em>
<em><span> </span></em>
<em> </em>
3) The <code><span>kernel.controller</span>
Event
kernel.controller事件是在controller被执行前初始化一些信息或者改变controller对象。
被调用的controller确定之后,HttpKernel::handle()就会调度kernel.controller事件。在系统的某部分被确定后(例如:controller、路由信息等)但是这些部分被执行前,监听kernel.controller事件的监听器就会运行了。
4)获得controller的参数
getAttributes()方法是返回一个参数数组,这个参数数组会被传递给controller,我们也可以自定义该方法,也可以使用Symfony框架内置的。
ControllerResolver使用放射机制获得被调用的controller的方法的参数列表。遍历该列表,使用下面
的步骤来确定参数列表中一一对应的值:
<code><code><span> </span></code></code>
<span><em>a) 使用参数作为键查找Request对象中的attributes数组,如果找到,那么相应的值会传入到controller的方法中,例如:controller方法的第一个参数是$name,那么在Request的attributes数组中包含$</em></span><em>attributes['name']的值,那么</em><span><em>$</em></span><em>attributes['name']就会被使用。</em>
<em> </em>
<em>b) 如果该该参数在Symfony配置routing的时候被指定,那么就会跳过对该参数的查找。</em>
<em> </em>
5)调用controller
这一步,controller就会被执行。
controller会创建包含特定页面或者json的Response对象,这也是应用层的最后一个步骤。
Symfony框架中没有缺省的监听器实现kernel.view事件,可是,有一个核心Bundle——SensioFrameworkExtraBundle里有个监听改事件的监听器。如果你的controller返回一个数组,并且在controller类的顶部有@Template的注解,那么该监听器就会渲染一个模板,把controller返回的数组传入到模板中,最后利用模板返回的内容创建一个Response对象,并返回该Response对象。
除此之外,FOSRestBundle也实现了监听该事件的监听器,a listener on this event which aims to give you a robust view layer capable of using a single controller to return many different content-type responses (e.g. HTML, JSON, XML, etc).
7) kernel.response 事件
在发送Response对象到客户端前修改它。
kernel的目的是把Request对象转换成为Response对象。Response对象可能是在kernel.request事件中创建,可能是由controller返回,又或者是由监听kernel.view事件的监听器返回。
不管是在哪一个环节创建Response对象,最后kernel.response事件都会被触发。监听kernel.response事件的监听器都会以某种方式修改Response对象,例如:修改Response的header部分,修改cookie,或者甚至会修改Response对象返回的内容(注入javascript到 Déclaration:Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn