Symfony2源码分析——启动过程2,symfony2源码_PHP教程
Symfony2源码分析——启动过程2,symfony2源码
上一篇分析Symfony2框架源码,探究Symfony2如何完成一个请求的前半部分,前半部分可以理解为Symfony2框架为处理请求做准备工作,包括container生成、缓存、bundls初始化等一些列准备工作(Symfony2源码分析——启动过程1)。而这一篇讲的是Symfony2如何根据请求的数据生成Response对象,向客户端返回响应数据。
在分析前需要了解Symfony2的事件驱动机制:Symfony2事件驱动。
言归正传,Symfony2请求的工作流程其实是Symfony2内核的事件驱动完成的,下面是Symfony2框架定义好的内核事件:

我们可以编写事件监听器,监听相应的内核事件,在Symfony2触发该事件的时候,相应的事件监听器就会执行。监听和唤醒形象的描述,就像,你(事件监听器)参加校运会,去大会(Symfony2)登记(监听)参加50米短跑(事件),当50米短跑比赛开始了(事件被触发),那你就奔跑吧(监听器执行,其实就是一个执行函数,函数完成什么工作就取决于你的需求了),少年。
Symfony2的内核事件处理流程大部分工作都在HttpKernel::handleRaw方法中:
<span> 1</span> <span>private</span> function handleRaw(Request $request, $type =<span> self::MASTER_REQUEST) </span><span> 2</span> <span> { </span><span> 3</span> $<span>this</span>->requestStack-><span>push($request); </span><span> 4</span> <span> 5</span> <span>//</span><span> request </span><span> 6</span> <span>//</span><span> 初始化事件,事件对象会被传递给监听器,所以事件可以说是一个信息的载体,事件内存放着监听器感兴趣的数据。</span> <span> 7</span> $<span>event</span> = <span>new</span> GetResponseEvent($<span>this</span><span>, $request, $type); </span><span> 8</span> <span>//</span><span> 触发kernel.request事件,后续详细讲解EventDispatcher::dispatch方法的实现, </span><span> 9</span> <span>//</span><span> 这里我们需要知道的是,dispatcher把$event传递给所有监听了kernel.request事件的监听器,监听器将会执行。 </span><span>10</span> <span>//</span><span> kernel.request事件发生在controller执行之前,我们可以在这一步奏完成路由解析等为controller执行提供准备数据, </span><span>11</span> <span>//</span><span> 在这个过程允许我们直接生成Response对象,向客户端输出数据,那么controller就不会被执行了。</span> <span>12</span> $<span>this</span>->dispatcher->dispatch(KernelEvents::REQUEST, $<span>event</span><span>); </span><span>13</span> <span>14</span> <span>//</span><span> 如果我们在kernel.request事件生成了Response对象(响应数据),那么就跳过kernel.controller、kernel.view事件、 </span><span>15</span> <span>//</span><span> controller也会被跳过,直接执行kernel.response事件。</span> <span>16</span> <span>if</span> ($<span>event</span>-><span>hasResponse()) { </span><span>17</span> <span>return</span> $<span>this</span>->filterResponse($<span>event</span>-><span>getResponse(), $request, $type); </span><span>18</span> <span> } </span><span>19</span> <span>20</span> <span>//</span><span> load controller </span><span>21</span> <span>//</span><span> 根据路由规则返回 一个对象或者数组或者字符串 ,如果$controller是一个数组,$controller[0]是存放的是要执行的controller对象, </span><span>22</span> <span>//</span><span> $controller[0]存放的是controller对象执行的方法,即action,方法的参数没有保存在$controller数组中; </span><span>23</span> <span>//</span><span> 如果$controller是对象,那么该对象就实现了__invoke 方法; </span><span>24</span> <span>//</span><span> 如果$controller是字符串,那么$controller就是要运行的函数的函数名。 </span><span>25</span> <span>//</span><span> 图2是$controller的一个var_dump例子</span> <span>26</span> <span>if</span> (<span>false</span> === $controller = $<span>this</span>->resolver-><span>getController($request)) { </span><span>27</span> <span>throw</span> <span>new</span> NotFoundHttpException(sprintf(<span>'</span><span>Unable to find the controller for path "%s". Maybe you forgot to add the matching route in your routing configuration?</span><span>'</span>, $request-><span>getPathInfo())); </span><span>28</span> <span> } </span><span>29</span> <span>30</span> $<span>event</span> = <span>new</span> FilterControllerEvent($<span>this</span><span>, $controller, $request, $type); </span><span>31</span> <span>//</span><span> 触发kernel.controller事件,这个事件发生在controller执行前。我们可以通过监听这个事件在controller执行前修改controller, </span><span>32</span> <span>//</span><span> 或者完成一些动作。</span> <span>33</span> $<span>this</span>->dispatcher->dispatch(KernelEvents::CONTROLLER, $<span>event</span><span>); </span><span>34</span> $controller = $<span>event</span>-><span>getController(); </span><span>35</span> <span>36</span> <span>//</span><span> controller arguments </span><span>37</span> <span>//</span><span> 从request对象中获取controller方法的参数</span> <span>38</span> $arguments = $<span>this</span>->resolver-><span>getArguments($request, $controller); </span><span>39</span> <span>40</span> <span>//</span><span> call controller </span><span>41</span> <span>//</span><span> 执行controller</span> <span>42</span> $response =<span> call_user_func_array($controller, $arguments); </span><span>43</span> <span>44</span> <span>//</span><span> view </span><span>45</span> <span>//</span><span> 如果$response不是Response对象,那么kernel.view事件就会触发,监听kernel.view事件的监听器通过$response值生成Response对象。</span> <span>46</span> <span>if</span> (!<span>$response instanceof Response) { </span><span>47</span> $<span>event</span> = <span>new</span> GetResponseForControllerResultEvent($<span>this</span><span>, $request, $type, $response); </span><span>48</span> $<span>this</span>->dispatcher->dispatch(KernelEvents::VIEW, $<span>event</span><span>); </span><span>49</span> <span>50</span> <span>if</span> ($<span>event</span>-><span>hasResponse()) { </span><span>51</span> $response = $<span>event</span>-><span>getResponse(); </span><span>52</span> <span> } </span><span>53</span> <span>54</span> <span>if</span> (!<span>$response instanceof Response) { </span><span>55</span> $msg = sprintf(<span>'</span><span>The controller must return a response (%s given).</span><span>'</span>, $<span>this</span>-><span>varToString($response)); </span><span>56</span> <span>57</span> <span>//</span><span> the user may have forgotten to return something</span> <span>58</span> <span>if</span> (<span>null</span> ===<span> $response) { </span><span>59</span> $msg .= <span>'</span><span> Did you forget to add a return statement somewhere in your controller?</span><span>'</span><span>; </span><span>60</span> <span> } </span><span>61</span> <span>throw</span> <span>new</span><span> \LogicException($msg); </span><span>62</span> <span> } </span><span>63</span> <span> } </span><span>64</span> <span>65</span> <span>//</span><span> 触发kernel.response事件,在向客户端输出Response对象前,我们可以对Response对象进行修改, </span><span>66</span> <span>//</span><span> 例如修改response头部,设置缓存、压缩输出数据等。 </span><span>67</span> <span>68</span> <span>//</span><span> 接着触发kernel.finish_request事件,把当前请求从请求栈中弹出,当前请求就完成。</span> <span>69</span> <span>return</span> $<span>this</span>-><span>filterResponse($response, $request, $type); </span><span>70</span> <span>71</span> <span>//</span><span> 千万别忘记了,filterResponse执行完后,Symfony2内核事件处理流程还有最后一步,位于app_dev.php[app.php]最后一行, </span><span>72</span> <span>//</span><span> $kernel->terminate($request, $response);这个方法触发kernel.terminate事件,此时,Symfony2已经响应了客户端的请求, </span><span>73</span> <span>//</span><span> 向客户端输出了Response对象。监听kernel.terminate事件的监听器,主要是为了完成一些耗时的操作,操作的结果不需要返回给 </span><span>74</span> <span>//</span><span> 客户端的,例如邮件发送、图片压缩等等。 </span><span>75</span> <span>//</span><span> 到这里,Symfony2的整个流程就走完了。</span> <span>76</span> }
<span>HttpKernel::filterResponse方法和</span>HttpKernel::finishRequest方法:


图2
Symfony2框架的事件分发机制的核心代码:
<span> 1</span> <span>public</span> function dispatch($eventName, Event $<span>event</span> = <span>null</span><span>) </span><span> 2</span> <span> { </span><span> 3</span> <span>if</span> (<span>null</span> === $<span>event</span><span>) { </span><span> 4</span> $<span>event</span> = <span>new</span><span> Event(); </span><span> 5</span> <span> } </span><span> 6</span> <span> 7</span> $<span>event</span>->setDispatcher($<span>this</span><span>); </span><span> 8</span> $<span>event</span>-><span>setName($eventName); </span><span> 9</span> <span>10</span> <span>if</span> (!isset($<span>this</span>-><span>listeners[$eventName])) { </span><span>11</span> <span>return</span> $<span>event</span><span>; </span><span>12</span> <span> } </span><span>13</span> <span>14</span> <span>//</span><span> $eventName即:KernelEvents::REQUEST、KernelEvents::CONTROLLER、KernelEvents::VIEW、KernelEvents::RESPONSE、KernelEvents::TERMINATE等 </span><span>15</span> <span>//</span><span> getListeners返回所有监听$eventName事件的监听器</span> <span>16</span> $<span>this</span>->doDispatch($<span>this</span>->getListeners($eventName), $eventName, $<span>event</span><span>); </span><span>17</span> <span>18</span> <span>return</span> $<span>event</span><span>; </span><span>19</span> <span> } </span><span>20</span> <span>21</span> <span>protected</span> function doDispatch($listeners, $eventName, Event $<span>event</span><span>) </span><span>22</span> <span> { </span><span>23</span> <span>//</span><span> 监听器执行</span> <span>24</span> <span>foreach</span> ($listeners <span>as</span><span> $listener) { </span><span>25</span> call_user_func($listener, $<span>event</span>, $eventName, $<span>this</span><span>); </span><span>26</span> <span>//</span><span> 如果其中一个监听器把$event的propagationStopped属性设置为true,那么表示$eventName这一事件终止执行, </span><span>27</span> <span>//</span><span> 事件不会往$listeners里尚未执行的监听器传递该事件。</span> <span>28</span> <span>if</span> ($<span>event</span>-><span>isPropagationStopped()) { </span><span>29</span> <span>break</span><span>; </span><span>30</span> <span> } </span><span>31</span> <span> } </span><span>32</span> }
我个人感觉毛德操先生的书linux内核情景分析,分析的很透彻的,对整个Linux内核的讲解很不错.具体到启动过程,这个启动过程很泛的,你只需要知道整体的启动过程,其他的细节,还得看内核源码讲解,所以,推荐你看点博客了解下启动大体过程,具体到每个细节,看linux内核情景分析吧.
1.这个dialog是由PhoneWindowManager控制的,在PhoneWindowManager的interceptKeyTq方法中,代码是这一行
mHandler.postDelayed(mPowerLongPress, ViewConfiguration.getGlobalActionKeyTimeout());
mProwerLongPress是一个Runnable,执行时调用GlobalActions中的showDialog方法。所以这个dialog是由GlobalActions管理的,PowerDialog是之前版本中的,现在已经弃用了。
2.可以在GlobalActions中createDialog方法中mSilentModeToggle action的onToggle方法中加入
mAudioManager.setVibrateSetting(AudioManager.VIBRATE_TYPE_RINGER, on ? AudioManager.VIBRATE_SETTING_ON : AudioManager.VIBRATE_SETTING_OFF);
这一句,仿照铃声的处理,应该没问题,不过没试所以也不能确定。
希望对你有帮助。

핫 AI 도구

Undresser.AI Undress
사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover
사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool
무료로 이미지를 벗다

Clothoff.io
AI 옷 제거제

AI Hentai Generator
AI Hentai를 무료로 생성하십시오.

인기 기사

뜨거운 도구

메모장++7.3.1
사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전
중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기
강력한 PHP 통합 개발 환경

드림위버 CS6
시각적 웹 개발 도구

SublimeText3 Mac 버전
신 수준의 코드 편집 소프트웨어(SublimeText3)

뜨거운 주제











운영 체제의 핵심 부분인 Linux 커널은 하드웨어 자원 관리 및 시스템 호출 제공과 같은 중요한 기능을 담당합니다. 이 기사에서는 프로세스 관리, 파일 시스템, 네트워크 통신, 장치 드라이버 및 메모리 관리를 포함하여 Linux 커널의 다섯 가지 주요 부분을 살펴보고 자세한 소개와 코드 예제를 제공합니다. 1. 프로세스 관리 프로세스 생성 Linux 커널에서 프로세스 생성은 fork() 시스템 호출을 통해 구현됩니다. 다음은 간단한 예제 코드입니다: #include

내용이 길고 기술적인 내용이 많기 때문에 클릭해서 따라가시면 길을 잃을 염려가 없습니다. 서문: Linux 커널 이해하기 컴퓨터 시스템은 하드웨어와 소프트웨어의 공생체이며 상호 의존적이며 분리될 수 없습니다. 컴퓨터 하드웨어 Linux 커널 이식 단계에는 컴퓨터 실린더를 구성하는 주변 장치, 프로세서, 메모리, 하드 드라이브 및 기타 전자 장치가 포함됩니다. 그리고 이를 작동하고 제어하는 소프트웨어가 없으면 자체적으로 작동할 수 없습니다. 이러한 제어 작업을 완료하는 소프트웨어를 Linux 용어로 "커널" 또는 "코어"라고 합니다. Linux 커널의 주요 모듈(또는 구성 요소)은 스토리지 관리, CPU 및 프로세스 관리, 파일 시스템, 장치 관리 및 드라이버, 네트워크 통신 Linux 포럼, 시스템 등의 부분으로 나뉩니다.

안녕하세요, 독자 여러분! 여기서 저는 Linux 커널 TCP 프로토콜 스택의 개발 및 최적화 분야에서 전문적인 기술을 바탕으로 수석 네트워크 엔지니어로서 축적한 귀중한 경험과 기술을 여러분과 공유하게 된 것을 영광으로 생각합니다. 이 글을 통해 우리는 서로 배우고 토론할 수 있으며, 이 분야에 큰 관심을 가지고 있거나 관련 작업을 하고 있는 여러분에게 실용적이고 유용한 참고 자료를 제공할 수 있다고 믿습니다. 1. TCP 연결 설정 TCP 연결 설정은 TCP 프로토콜 스택의 핵심 트랜잭션이지만 많은 연결 문제에 직면하는 것은 드문 일이 아닙니다. 신중하게 고려하고 세부적으로 디버깅한 후 SYN 플러딩 공격 방지(시스템 매개변수 조정) 및 네트워크 정체 처리(즉, TCPFastOp 사용)를 포함하여 몇 가지 일반적이고 실제적인 문제와 해결 방법을 발견했습니다.

이전 기사에서는 RISC-V Linux 시작을 위한 페이지 테이블 생성을 분석했습니다. RISC-V Linux 항목 주소는 2M 정렬되어야 한다고 언급되었습니다. 오늘은 2M 정렬 문제를 해결하는 방법이나 일부를 최적화하는 방법에 대해 설명하겠습니다. 기억.

이것은 Linux 커널 소스 코드 배포를 심층적으로 탐구하는 1500 단어 분량의 기사입니다. 제한된 공간으로 인해 Linux 커널 소스 코드의 조직 구조에 중점을 두고 독자의 이해를 돕기 위해 몇 가지 구체적인 코드 예제를 제공합니다. Linux 커널은 소스 코드가 GitHub에서 호스팅되는 오픈 소스 운영 체제 커널입니다. 전체 Linux 커널 소스 코드 배포판은 매우 방대하며 여러 하위 시스템 및 모듈과 관련된 수십만 줄의 코드를 포함합니다. Linux 커널 소스 코드를 더 깊이 이해하려면

Linux 커널이 컴퓨터 운영 체제에서 중요한 역할을 한다는 관점을 논의합니다. Linux 커널 설계 및 실제 응용 프로그램에 대한 심층적인 분석을 통해 이 분야에서 Linux 커널의 탁월한 위치와 영향력을 드러냅니다. 1. 최적화된 메모리 관리 가상 메모리 관리 기술을 사용하여 Linux 커널은 메모리 할당 및 재활용을 효율적으로 완료할 수 있습니다. 교체 페이지 알고리즘의 도움으로 Linux 커널은 물리적 메모리와 가상 메모리 간의 매핑 관계를 정확하게 처리하도록 설계 및 구현되었습니다. 애플리케이션의 특정 요구 사항에 따라 유연하게 조정할 수 있으므로 전체 시스템 성능이 향상됩니다. 2. 강력한 프로세스 관리 커널은 뛰어난 멀티 태스킹 기술을 사용하여 단일 시스템에서 여러 프로세스가 조화롭게 공존할 수 있도록 합니다. 신중하게 구성됨

Android 시스템과 Linux 커널은 밀접하게 관련된 두 개체이며, 둘 사이의 관계는 밀접하고 복잡합니다. Android 시스템에서 Linux 커널은 Android 시스템에 대한 기본 하드웨어 드라이버 및 시스템 호출 지원을 제공하는 중요한 역할을 합니다. 이 기사에서는 Android 시스템과 Linux 커널 간의 관계, 상호 작용 및 작동 방식을 살펴보고 몇 가지 구체적인 코드 예제를 제공합니다. 안드로이드(Android)는 리눅스 커널을 기반으로 개발된 모바일 운영체제로 스마트폰, 태블릿 등 모바일 기기에 주로 사용된다. 엘

Linux 커널은 운영 체제의 핵심이며 CPU, I/O 장치, 물리적 메모리 및 파일 시스템과 같은 시스템 리소스에 대한 액세스를 제어합니다. 부팅 과정과 시스템이 실행되는 동안 커널은 커널 링 버퍼에 다양한 메시지를 씁니다. 이러한 메시지에는 시스템 작동에 대한 다양한 정보가 포함됩니다.
