原文:http://www.blogjava.net/AllanZ/archive/2009/07/20/287472.html
这总览,很明显是看FacesServlet的service方法。在FacesServlet的初始化过程中,构造出了全局的FacesContextFactory对象和LifeCycle对象。可以把FacesContextFactory看做是一个“请求包装工厂”,于是很明显,每当一个请求到达FacesServlet的时候,第一步便是拿着请求,到包装工厂里面包装一下,而包装的结果就是一个FacesContext。代码如下:
1 FacesContext context = facesContextFactory.getFacesContext(servletConfig.getServletContext(), request, response, lifecycle);
在包装过程中,实际上是创建了一个com.sun.faces.context.FacesContextImpl对象,FacesContextImpl类继承了jsf-api项目中的javax.faces.context.FacesContext。FacesContextImpl的构造方法的第一个参数是一个叫做ExternalContext的接口的实现,查看其源代码,可以看到ExternalContextImpl类耦合了Servlet API,而FacesContextImpl与Servlet API无关。实际上,在这里,做到了JSF可以不仅仅使用在Servlet环境中,正如ExternalContext接口的注释中所说,在Servlet环境中使用JSF和在Portlet环境中使用JSF的不同,实际上就是使用了不同的ExternalContext。在FacesContextFactoryImpl中构造FacesContextImpl的代码如下:
<!--<br />
<br />
Code highlighting produced by Actipro CodeHighlighter (freeware)<br />
http://www.CodeHighlighter.com/<br />
<br />
-->1 FacesContext ctx = new FacesContextImpl(new ExternalContextImpl((ServletContext) sc,(ServletRequest) request,(ServletResponse) response),lifecycle);
FacesContextImpl的构造方法中,还做了另外一件事情,就是根据配置确定了RenderKitFactory,显然不同的RenderKitFactory可以产生不同的RenderKit,而不同RenderKit对象是针对不同客户端的,所以对于浏览器、移动设备等等,会有不同的RenderKit。FacesContextImpl的构造方法中代码如下:
<!--<br />
<br />
Code highlighting produced by Actipro CodeHighlighter (freeware)<br />
http://www.CodeHighlighter.com/<br />
<br />
-->1 this.externalContext = ec;
2 setCurrentInstance(this);
3 this.rkFactory = (RenderKitFactory)FactoryFinder.getFactory(FactoryFinder.RENDER_KIT_FACTORY);
在代码中我们经常使用FacesContext.getCurrentInstance()这个静态方法来获取与当前请求对应的FacesContext对象,实际上是在FacesContext类里面有一个静态的ThreadLocal对象用来存放了当前请求线程对应的FacesContext对象,于是上面的代码中setCurrentInstance(this)就是把当前构造出来的这个FacesContext对象放到了ThreadLocal里面。
FacesContext创建出来以后,正如上面所说,要让他经过LifeCycle这个“Filter Chain”的逐步处理了。那么,Filter Chain里面放的是一个一个Filter,那么LifeCycle这个Chain里面放的是什么呢?答案是Phases。
FacesServlet让FaceContext通过LifeCycle的处理,分成了两个部分。一个部分是调用LifeCycle的execute方法,执行逻辑,第二个部分是调用LifeCycle的render方法,呈现响应。FacesServlet.service中代码如下:
<!--<br />
<br />
Code highlighting produced by Actipro CodeHighlighter (freeware)<br />
http://www.CodeHighlighter.com/<br />
<br />
-->1 lifecycle.execute(context);
2 lifecycle.render(context);
在LifeCycleImpl这个实现中,存放了一个Phase对象的数组,存放了7个Phase。其中第一个是null,然后依次是视图重建、应用请求值、验证、更新模型值、执行应用程序、呈现响应。在execute方法中,调用了从视图重建开始到执行应用程序为止的5个Phase,而在render方法中,调用了最后一个Phase,也就是呈现响应。在LifeCycleImpl类中,代码如下:
<!--<br />
<br />
Code highlighting produced by Actipro CodeHighlighter (freeware)<br />
http://www.CodeHighlighter.com/<br />
<br />
--> //The Phase instance for the render() method
private Phase response = new RenderResponsePhase();
// The set of Phase instances that are executed by the execute() method
// in order by the ordinal property of each phase
private Phase[] phases = {
null, // ANY_PHASE placeholder, not a real Phase
new RestoreViewPhase(),
new ApplyRequestValuesPhase(),
new ProcessValidationsPhase(),
new UpdateModelValuesPhase(),
new InvokeApplicationPhase(),
response
};
在Servlet Filter中,可以由每一个Filter来决定是否要调用下一个Filter,从而决定是否让请求继续通过Filter Chains中的后续Filter,是链式调用的过程。而在LifeCycle的execute方法中,是用一个for循环顺序执行几个Phase。在每一个Phase执行完之后,都会检查FaceContext对象中是否设置了停止后续处理直接呈现响应的标志(renderResponse)或者已经完成了响应无需后续处理也不需要经过呈现响应阶段了(responseComplete),如果标志为true,那么就不再执行后续Phase。
LifeCycleImpl的execute方法主要代码如下:
<!--<br />
<br />
Code highlighting produced by Actipro CodeHighlighter (freeware)<br />
http://www.CodeHighlighter.com/<br />
<br />
--> 1 for (int i = 1, len = phases.length -1 ; i < len; i++) { // Skip ANY_PHASE placeholder
2
3 if (context.getRenderResponse() ||
4 context.getResponseComplete()) {
5 break;
6 }
7
8 phases[i].doPhase(context, this, listeners.listIterator());
9
10 }
在LifeCycle的render方法中,也会检查FacesContext的responseComplete状态,如果为true,那么就不再执行render Phase。于是我们此刻知道了在我们自己所写的一些代码或者JSF库里面的一些代码中,调用FacesContext的responseComplete方法和renderResponse得作用原理。render方法主要代码如下:
<!--<br />
<br />
Code highlighting produced by Actipro CodeHighlighter (freeware)<br />
http://www.CodeHighlighter.com/<br />
<br />
-->1 if (!context.getResponseComplete()) {
2 response.doPhase(context, this,listeners.listIterator());
3 }
另外注意,Phase这个概念、接口,以及几个实现,都是jsf-ri项目中的,而在jsf-api中不存在Phase这个概念。所以,LifeCycle是JSF标准的内容,而通过几个Phase来处理请求这种实现是sun的参考实现的做法。
最后,可以看到对于每一个phase都调用了doPhase方法,同时把LifeCycle和FacesContext当做参数传入了。值得注意的是,所谓的phaseListener,也传入了phase的doPhase方法中,由此大约能够想明白这个“阶段监听器”的道理了。
分享到:
相关推荐
JSF请求处理生命周期图
- **前端控制器 FacesServlet**:这是一个由JSF提供的类,用于处理所有针对JSF的请求。在`web.xml`文件中配置此控制器,启动请求处理的生命周期。 - **Faces配置文件**:默认配置文件位于`WEB-INF/faces-config.xml`...
JSF的每个请求都经历一个预定义的“请求处理生命周期”,分为若干阶段,包括初始化、应用请求值、过程验证、更新模型值、调用应用、呈现响应等。这个生命周期根据不同的请求类型有不同的执行路径,例如: 1. **非...
4. **生命周期**:JSF有自己的一套请求处理生命周期,包括初始化、应用请求值、过程验证、更新模型值、调用应用逻辑和渲染响应等阶段。JSF 1.2优化了这个生命周期,提供了更多的控制点和错误处理机制。 5. **...
在这个过程中,JSF处理用户提交的数据,进行验证,更新模型,然后渲染响应。 ### PPT演示 PPT演示通常包含详细的教程,涵盖了JSF的基本概念、组件使用、页面导航、数据绑定、事件处理、错误处理等方面。这些资源...
- **控制器(Controller)**:JSF生命周期处理用户请求,执行验证、更新模型和调用业务逻辑。 ### JSF模型 JSF模型是应用程序的数据层,由管理Bean(Managed Beans)组成,它们是Java类,包含了应用程序的业务逻辑...
### JSF生命周期的事件处理 JavaServer Faces(简称JSF)是Java平台提供的一种用于构建企业级Web应用程序的标准框架。JSF通过定义一套简洁而强大的API来帮助开发者更轻松地开发用户界面。其中,JSF的生命周期管理...
JSF组件有六个主要阶段:恢复视图、应用请求值、处理验证、更新模型值、调用应用程序和渲染响应。在每个阶段,JSF都会执行特定的任务,如解析请求参数、执行验证规则、更新Bean属性等。 ### 自定义标签库 JSF允许...
事件驱动模型是JSF框架的灵魂,它将Web应用的交互过程转换为事件处理机制,极大地简化了开发者的工作。在JSF中,UI组件不仅可以作为事件的来源,也可以作为事件的监听者。例如,当用户点击按钮时,该按钮作为事件源...
9. **FacesServlet**: 这是JSF应用的核心Servlet,负责处理所有的JSF请求,并调用相应的生命周期阶段。 使用`jsf-api.jar`,开发者可以轻松地整合其他Java EE技术,如EJB(Enterprise JavaBeans)、JPA(Java ...
这个第二版很可能包含了JSF的最新版本特性,比如JSF 2.x版本引入的改进,如Facelets视图技术、异步请求处理和CDI集成等。 总的来说,《JSF核心编程第二版》是一本全面的教程,旨在帮助开发者熟练掌握JSF框架,从而...
JSF有六个阶段:恢复视图、应用请求值、处理验证、更新模型值、调用应用和渲染响应。在这些阶段中,JSF控件的值被初始化、验证、更新,并最终显示给用户。 ### JSF组件库 除了标准JSF组件外,还有许多第三方库提供...
JSF的生命周期包括多个阶段,如恢复视图、应用请求值、处理验证等。掌握每个阶段的作用及如何编写相关的事件处理代码是必要的。 ##### 2.4 一个JSF应用的实际案例 通过实际案例来学习JSF的最佳实践。比如开发一个...
#### 二、JSF架构组成 根据给定的内容,“JSF架构图zz”主要涉及到了JSF框架的几个关键组成部分,下面将详细介绍这些部分及其在JSF框架中的作用。 ##### 1. **表示层应用程序逻辑** 表示层是JSF框架的核心,负责...
例如,可以通过查看`FacesServlet`的源码了解JSF如何处理HTTP请求,或者研究`UIComponent`类以了解组件的生命周期和渲染过程。 **工具**在JSF开发中可能指的是IDE插件、构建工具或者调试工具。例如,Eclipse和...
2. **FacesServlet**:JSF的核心是FacesServlet,它是处理所有JSF请求的入口点。当用户与JSF页面交互时,请求会被转发到FacesServlet,该Servlet负责解析请求,执行相应的动作,更新模型,并呈现响应。 3. **...
在JSF中,`FacesServlet`扮演控制器的角色,解析请求并调用相应的后台处理方法。 ### JSF组件 JSF组件是UI元素,如按钮、输入框、表格等。每个组件都有一个标识符,可以通过XML(Facelets)或者Java代码声明在页面...
在深入理解JSF请求处理生命周期之前,我们先要明白JSF的核心目标是简化Web开发,尤其是处理用户输入和更新服务器端状态的过程。生命周期的概念在JSF中扮演着至关重要的角色,因为它自动化了许多传统Web技术中需要...