Yaf支持用户定义插件来扩展Yaf的功能, 这些插件都是一些类. 它们都必须继承自Yaf_Plugin_Abstract
.
插件要发挥功效, 也必须现实的在Yaf中进行注册, 然后在适当的实际, Yaf就会调用它.
也许大家会问这个插件是个什么概念,有什么用呢。
其实我们用插件主要是用到Yaf框架中支持的Hook(钩子),Yaf中定义了7个Hook。
1 Yaf中支持的7个Hook
Yaf的工作流程如下,7个Hook插在流程的不同位置。
routerStartup
:这个会在路由之前出发,也就是路由之前会调用这个Hook ,这个是7个事件中, 最早的一个. 但是一些全局自定的工作, 还是应该放在Bootstrap中去完成routerShutdown
:这个在路由结束之后触发,需要注意的是,只有路由正确完成之后才会触发这个HookdispatchLoopStartup
:分发循环开始之前被触发preDispatch
:分发之前触发,如果在一个请求处理过程中, 发生了forward
, 则这个事件会被触发多次postDispatch
:分发结束之后触发,此时动作已经执行结束, 视图也已经渲染完成. 和preDispatch
类似, 此事件也可能触发多次dispatchLoopShutdown
:分发循环结束之后触发 此时表示所有的业务逻辑都已经运行完成, 但是响应还没有发送preResponse
:响应之前触发
其实挺希望有一个render之前的Hook的,可惜Yaf没有。
2 插件用法
以上只是对插件做了一个基本的介绍,如果此前没有接触过类似设计的可能一下子稀里糊涂的。
其实也没关系,当前最重要的就是记住,有这么个印象,在后续我们使用的过程当中慢慢的就能接受了。
插件类是用户编写的, 但是它需要继承自Yaf_Plugin_Abstract
.
对于插件来说, 上一节提到的7个Hook, 不需要全部关心。
只需要在插件类中定义和上面事件同名的方法, 这个方法就会在该事件触发的时候被调用.
而插件方法, 可以接受俩个参数, Yaf_Request_Abstract
实例和Yaf_Response_Abstract
实例.
2.1 编写插件
一个插件类例子如下:
<?php class UserPlugin extends Yaf_Plugin_Abstract { public function routerStartup(Yaf_Request_Abstract $request, Yaf_Response_Abstract $response) { } public function routerShutdown(Yaf_Request_Abstract $request, Yaf_Response_Abstract $response) { } }
2.2 注册插件
插件要生效, 还需要向Yaf_Dispatcher
注册, 那么一般的插件的注册都会放在Bootstrap
中进行.
一个注册插件的例子如下:
<?php class Bootstrap extends Yaf_Bootstrap_Abstract{ public function _initPlugin(Yaf_Dispatcher $dispatcher) { $user = new UserPlugin(); $dispatcher->registerPlugin($user); } }
2.3 插件目录
一般的, 插件应该放置在APPLICATION_PATH
下的plugins
目录。
这样在自动加载的时候, 加载器通过类名, 发现这是个插件类, 就会在这个目录下查找.
当然, 插件也可以放在任何你想防止的地方, 只要你能把这个类加载进来就可以。
3 完整案例
这就是插件的使用过程,对于上面我们做一个总结,用代码来说话。
首先,我们定义好自己的插件类:
/** * 插件类定义 * UserPlugin.php */ class UserPlugin extends Yaf_Plugin_Abstract { //在路由之前触发,这个是7个事件中, 最早的一个. 但是一些全局自定的工作, 还是应该放在Bootstrap中去完成 public function routerStartup(Yaf_Request_Abstract $request, Yaf_Response_Abstract $response) { echo "Plugin routerStartup called <br/>\n"; } //路由结束之后触发,此时路由一定正确完成, 否则这个事件不会触发 public function routerShutdown(Yaf_Request_Abstract $request, Yaf_Response_Abstract $response) { echo "Plugin routerShutdown called <br/>\n"; } //分发循环开始之前被触发 public function dispatchLoopStartup(Yaf_Request_Abstract $request, Yaf_Response_Abstract $response) { echo "Plugin DispatchLoopStartup called <br/>\n"; } //分发之前触发 如果在一个请求处理过程中, 发生了forward, 则这个事件会被触发多次 public function preDispatch(Yaf_Request_Abstract $request, Yaf_Response_Abstract $response) { echo "Plugin PreDispatch called <br/>\n"; } //分发结束之后触发,此时动作已经执行结束, 视图也已经渲染完成. 和preDispatch类似, 此事件也可能触发多次 public function postDispatch(Yaf_Request_Abstract $request, Yaf_Response_Abstract $response) { echo "Plugin postDispatch called <br/>\n"; } //分发循环结束之后触发,此时表示所有的业务逻辑都已经运行完成, 但是响应还没有发送 public function dispatchLoopShutdown(Yaf_Request_Abstract $request, Yaf_Response_Abstract $response) { echo "Plugin DispatchLoopShutdown called <br/>\n"; } public function preResponse(Yaf_Request_Abstract $request, Yaf_Response_Abstract $response) { echo "Plugin PreResponse called <br/>\n"; } }
然后注册我们的插件,在Bootstrap注册插件
class Bootstrap extends Yaf_Bootstrap_Abstract{ /** * 注册一个插件 * 插件的目录是在application_directory/plugins */ public function _initPlugin(Yaf_Dispatcher $dispatcher) { $user = new UserPlugin(); $dispatcher->registerPlugin($user); } }
就这样,插件就会在我们的项目运行过程中自动调用相关的Hook,我们可以在这些Hook中部署自己的业务逻辑。
比如用户是否需要登录,权限判断等。
参考资料: