• Symfony2 学习笔记之内部构件



      Symfony2内部是怎样工作的以及我们如何来扩展它呢?
      从外部整体上看,symfony2代码是由许多独立的层构成,每一层都是建立在前一层基础之上。其中,自动加载时不受框架直接管理的,它完全是在UniversalClassLoader类和src/autoload.php文件的帮助下独立完成的。

    HttpFoundation 组件
      最深层次的是HttpFoundation组件,它提供了处理HTTP所需的主要对象。是一个对一些PHP函数和变量的面向对象抽象。
    包括:
      Request 类,抽象了PHP中主要的全局变量$_GET,$_POST,$_COOKIE,$_FILES 和 $_SERVER。
      Response类,抽象类一些PHP函数比如header(), setcookie()和echo();
      Session 类和SessionStorageInterface接口则是抽象了Session管理Session_*()函数。

    HttpKernel 组件:
      在HttpFoundation组件之上创建的一个组件,它处理HTTP的动态部分。它是为了能够通过标准的方式来处理request,而对Request和Response对象的一个最小封装。同时它提供了一些扩展点和工具,让它成为创建一个web框架的最理想的开始点。

      另外,Dependency Injection组件和强大的插件系统bundles让它增加了可配置性和扩展性。

    FrameworkBundle Bundle
      FrameworkBundle bundle 是一个bundle,它是构成轻量级快速MVC框架的主要组件和类库。

    Kernel
      HttpKernel类是Symfony2的中心类,它负责处理客户端请求。它的主要任务就是把Request对象转换成Response对象。每个Symfony2核心实现HttpKernelInterface接口。

    function handle(Request $request,$type=self::MASTER_REQUEST,$catch=true)


    Controller
      Kernel依靠Controller来吧Request转换为Response。 Controller可以是任何有效的PHP调用。Kernel委托选择哪个Controller应该被执行给一个ControllerResolverInterface接口的实现者。

    public function getController(Request $request);
    public function getArguments(Request $request,$controller);

      其中,getController()方法返回一个和给定的Request相对于的Controller(一个PHP调用)。ControllerResolver的默认实现是查找Request的一个_controller属性,它的值是一个class::method 格式的字符串。比如Bundle\BlogBundle\PostController::indexAction 。默认的实现使用RouterListener来定义Request的属性 _controller。getArguments()方法返回一个输入参数数组来传递给Controller调用。默认实现会根据Request属性自动的获取这些输入参数。

      从Request属性中匹配Controller方法的输入参数:Symfony2对于每一个方法的输入参数都会是这从Request中查找其同名属性,如果没有定义,就会取该输入参数的默认值。

    // Symfony2 从Request属性中查找 'id' 属性(强制)
    // 和一个'admin' 属性 (可选)
    public function showAction($id, $admin = true)
    {
           // ...
    }

    处理请求:handle()方法需要一个Request参数并且永远都必须返回一个Response。要转换Request,handle()需要依靠一个分析器和一个事件通知顺序链。
      1. 在处理任何事情之前,kernel.request事件将被通知 --如果一个监听者返回了一个Response,那么它会直接跳至第8步。
      2. 分析器被调用来判断哪个Controller将被执行。
      3. kernel.controller事件监听器现在开始处理Controller调用(改变它,封装它...)
      4. Kernel检查Controller是否是一个合法可调用的PHP回调。
      5. 分析器被调用来决定传递给controller的参数
      6. Kernel调用Controller
      7. 如果Controller没有返回Response对象,kernel.view事件监听器会把Controller的返回值转换成一个Response。
      8. kernel.response事件监听器开始处理Response(内容和头部);
      9. Response对象被返回。

      如果在这个过程中有一个异常被抛出,kernel.exception就会被通知,监听器就把异常转换为一个Response,之后kernel.response事件就会被通知,如果没能转换,该异常就会被抛出。

      如果你不想异常被捕获,你可以通过传递一个false作为第三个参数到handle()方法,来关闭kernel.exception事件。

    内部请求
      在处理一个主请求的任何时候,子请求都可以被处理。你可以传递一个请求类型到handle()方法,作为它的第二个参数。

    HttpKernelInterface::MASTER_ReQUEST;
    HttpKernelInterface::SUB_REQUEST

    这些类型会根据需要传递给所有的事件和监听器。


    事件
      Kernel抛出的每一个事件都会是KernelEvent类的子类。这就意味着每个事件都可以访问相同的基础信息。
        getRequestType() 返回请求的类型(HttpKernelInterface::MASTER_REQUEST 或者 HttpKernelInterface::SUB_REQUEST;
        getKernel() 返回处理请求的Kernel
        getRequest() 返回一个当前被处理的请求

      getRequestType()方法允许监听器知道请求的类型。比如,如果一个监听器必须是主请求才能激活它,你可以把该代码添加到你监听器方法的开头:

    use Symfony\Component\HttpKernel\HttpKernelInterface;
    if(HttpKernelInterface::MASTER_REQUEST !== $event->getRequestType()){
          //立刻返回
          return;
    }

    kernel.request 事件
    事件类:GetResponseEvent

      该事件的目标是立刻返回一个Response对象或者创建一个在事件结束后Controller可以调用的变量。任何监听器都可以通过event的setResponse()方法返回一个Response对象,当有Response对象被返回时,其它的监听器就不能在被调用了。

      FrameworkBundle使用事件通过RouterListener来发布一个_controller 请求属性。
      RequestListener 使用一个RouterInterface对象匹配Request,决定哪个Controller的名字会被存储到_controller的请求属性里。


    kernel.controller 事件:
    事件类: FilterControllerEvent

      FrameworkBundle不会使用该事件,但是该事件可以被作为一个修改要执行的controller的一个入口点。

    use Symfony\Component\Httpkernel\Event\FilterControllerEvent;
    
    public function onKernelController(FilterControllerEvent $event)
    {
            $controller = $event->getController();
             //...
    
            // 此处controller可以被该换成任何PHP可回调函数
            $event->setController($controller);
    }

    kernel.view 事件
    事件类:GetResponseForControllerResultEvent

      FrameworkBundle也不会使用该事件,但是它被用来实现一个视图子系统。该事件只有当Controller不能返回一个Response对象时才被调用。它的目的就是把其他类型的返回值转换成一个Response。

      Controller的返回值可以通过getControllerResult方法访问:

    use Symfony\Component\HttpKernel\Event\GetResponseForControllerResultEvent;
    use Symfony\Component\HttpFoundation\Response;
    
    public function onKernelView(GetResponseForControllerResultEvent $event)
    {
            $val = $event->getControllerResult();
            $response = new Response();
            //通过返回值自定义化Response
            $event->setResponse($response);
    }


    kernel.response 事件
    事件类:FilterResponseEvent

      该事件的目的是允许其它系统在Response对象被创建后对它进行修改或者替换。

    public function onKernelResponse(FilterResponseEvent $event)
    {
           $response = $event->getResponse();
           //修改Response对象
    }

    FrameworkBundle注册了许多的监听器:

            ProfilerListener 从当前请求中搜集数据
            WebDebugToolbarListener 注入Web 调试工具条
            ResponseListener 基于请求的格式来为Response设置Content-type
            EsiListener 当Response需要解析ESI标签时,向其添加一个Surrogate-Control HTTP头。

    kernel.exception 事件:
    事件类:GetResponseForExceptionEvent
         FrameworkBundle注册一个ExceptionListener把请求定向到一个给定Contoller。这个事件的监听器可以创建和设置一个Response对象,创建设置一个新的Exception对象或者什么都不做。

    use Symfony\Component\HttpKernel\Event\GetResponseForExceptionEvent;
    use Symfony\Component\HttpFoundation\Response;
    
    public function onKernelException(GetResponseForExceptionEvent $event)
    {
            $exception = $event->getException();
            $response = new Response();
            // 基于捕获的异常创建一个Response对象
            $event->setResponse($response);
    
            // 你可以创建一个新的异常代替原有的
            // $exception = new \Exception('Some special exception');
            // $event->setException($exception);
    }

    总结思考:我们了解了Symfony2内部的主要部件和一些主要的事件接口,我们可以在各个事件接口定义相应的监听器处理,来对请求处理过程进行干预操作。

    参考URL:http://symfony.com/doc/current/book/internals.html

  • 相关阅读:
    项目ITP(五) spring4.0 整合 Quartz 实现任务调度
    [Git 系列] WIN7下Git的安装
    Candy
    OSGI
    JAVA编程思想(1)
    [python] 字典和列表中的pop()函数
    R语言编程语法
    Linux 之创建工作目录-mkdir
    python 之 改变工作目录
    python 之 'and' 和 'or'
  • 原文地址:https://www.cnblogs.com/Seekr/p/2562780.html
Copyright © 2020-2023  润新知