JSF应用中,在web.xml里面配置了一个Servlet,叫做javax.faces.webapp.FacesServlet,于是可以知道,查看、了解一个请求的处理过程可以从这里开始。从官方网站上下载JSF的源代码,项目名比较古怪,叫做“mojarra”,我看的版本是1.2_12_b01。里面包含了两个子项目,一个是jsf-api,里面大多是接口以及少量关键类。另外一个项目叫做jsf-ri,对着这个"ri"邪念了半天之后,终于在兄弟提醒之下想明白了是reference implementation的意思。jsf-api是JavaEE标准的一部分,里面的类型包名都是以javax.faces开头的,而jsf-ri项目是sun针对JSF标准的一个参考实现,里面的类型的包名都是以com.sun.faces开头的。
FacesServlet初始化(FacesServlet#init)
JSF请求处理过程中,系统启动的时候,会初始化FacesServlet,调用其中的init方法。里面主要做了两件事情,一个是初始化FacesContextFactory,另外一个是初始化Lifecycle对象。在jsf-api项目中,FacesServlet类是一个Servlet接口的实现类,而FacesContextFactory和Lifecycle都是接口。在jsf-ri项目中有这两个接口的实现类,分别是com.sun.faces.context.FacesContextFactoryImpl和com.sun.faces.lifecycle.LifecycleImpl类。一个想当然的事实:FacesServlet初始化的时候要根据一些配置来判断具体的FacesContextFactory和Lifecycle实现类是什么,也就是在这里,“JSF标准”和“JSF实现”接轨了。想来MyFaces等等的其他JSF实现应该不外乎两种方式,一种是改变FacesServlet的init方法中需要用到的配置的值,于是启用自己的FacesContextFactory实现和Lifecycle实现,后面的处理过程就全部走自己的逻辑了。第二种方法笨一点,可能性不大,就是把FacesServlet覆盖替换掉,其中也不需要读什么配置了,直接使用自己的实现类即可——不过这种做法估计不符合JSF规范,想来只有我等虾米民众能做的出来。主要代码如下:
1 facesContextFactory = (FacesContextFactory)FactoryFinder.
getFactory(FactoryFinder.FACES_CONTEXT_FACTORY);
2
3 LifecycleFactory lifecycleFactory = (LifecycleFactory)FactoryFinder.
getFactory(FactoryFinder.LIFECYCLE_FACTORY);
回头再来看初始化的结果,FacesContextFactory很明显是用来生产FacesContext这么个东西的。而FacesContext可以看做是一个RequestWrapper(注意这个FaceContext和ServletContext不一样,ServletContext是一个Web应用只有一个的全局对象,对应的是一个Web application,而一个FacesContext对应的是一个request,另外,RequestWrapper这个说法不严格,实际上FacesContext里面也包装了ServletContext、Response等)。而LifeCycle可以看做是一个过滤器链(类似于servlet规范里面的Filter Chain)。于是,整个JSF请求处理过程,实际上就是包装成为FaceContext的用户请求,通过类似于一个Filter Chain的LifeCycle的过程。
这总览,很明显是看FacesServlet的service方法。在FacesServlet的初始化过程中,构造出了全局的FacesContextFactory对象和LifeCycle对象。可以把FacesContextFactory看做是一个“请求包装工厂”,于是很明显,每当一个请求到达FacesServlet的时候,第一步便是拿着请求,到包装工厂里面包装一下,而包装的结果就是一个FacesContext。代码如下:
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的代码如下:
FacesContext ctx = new FacesContextImpl
(new ExternalContextImpl((ServletContext) sc,
(ServletRequest) request,(ServletResponse) response),
lifecycle);
FacesContextImpl的构造方法中,还做了另外一件事情,就是根据配置确定了RenderKitFactory,显然不同的RenderKitFactory可以产生不同的RenderKit,而不同RenderKit对象是针对不同客户端的,所以对于浏览器、移动设备等等,会有不同的RenderKit。FacesContextImpl的构造方法中代码如下:
this.externalContext = ec;
setCurrentInstance(this);
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中代码如下:
lifecycle.execute(context);
lifecycle.render(context);
在LifeCycleImpl这个实现中,存放了一个Phase对象的数组,存放了7个Phase。其中第一个是null,然后依次是视图重建、应用请求值、验证、更新模型值、执行应用程序、呈现响应。在execute方法中,调用了从视图重建开始到执行应用程序为止的5个Phase,而在render方法中,调用了最后一个Phase,也就是呈现响应。在LifeCycleImpl类中,代码如下:
//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方法主要代码如下:
for (int i = 1, len = phases.length -1 ; i < len; i++) { // Skip ANY_PHASE placeholder
if (context.getRenderResponse() ||
context.getResponseComplete()) {
break;
}
phases[i].doPhase(context, this, listeners.listIterator());
}
在LifeCycle的render方法中,也会检查FacesContext的responseComplete状态,如果为true,那么就不再执行render Phase。于是我们此刻知道了在我们自己所写的一些代码或者JSF库里面的一些代码中,调用FacesContext的responseComplete方法和renderResponse得作用原理。render方法主要代码如下:
if (!context.getResponseComplete()) {
response.doPhase(context, this,listeners.listIterator());
}
另外注意,Phase这个概念、接口,以及几个实现,都是jsf-ri项目中的,而在jsf-api中不存在Phase这个概念。所以,LifeCycle是JSF标准的内容,而通过几个Phase来处理请求这种实现是sun的参考实现的做法。
最后,我们在JSF请求处理过程中可以看到对于每一个phase都调用了doPhase方法,同时把LifeCycle和FacesContext当做参数传入了。值得注意的是,所谓的phaseListener,也传入了phase的doPhase方法中,由此大约能够想明白这个“阶段监听器”的道理了。
分享到:
相关推荐
### JSF UI组件详解 #### 一、引言 JavaServer Faces (JSF) 是一个为简化企业级Java Web应用程序创建而设计的标准组件框架。它不仅提供了强大的模型-视图-控制器(MVC)架构,还引入了一系列标准UI组件,使得开发者...
在深入理解JSF请求处理生命周期之前,我们先要明白JSF的核心目标是简化Web开发,尤其是处理用户输入和更新服务器端状态的过程。生命周期的概念在JSF中扮演着至关重要的角色,因为它自动化了许多传统Web技术中需要...
**JSF 1.2核心详解** JavaServer Faces (JSF) 是Java平台上的一个用于构建用户界面的组件模型框架,特别适用于Web应用程序的开发。JSF 1.2是该框架的一个重要版本,它在JSF 1.1的基础上进行了一系列的改进和增强,...
### JSF框架架构详解 #### 一、JSF框架概述 JavaServer Faces(简称JSF)是一种基于Java的标准Web应用程序框架,它简化了基于组件的用户界面开发过程。JSF框架提供了一种易于使用的API来构建动态Web应用程序,并且...
### JSF+Spring+Hibernate整合图文教程详解 #### 一、多层体系结构与JSF+Spring+Hibernate的整合 **多层体系结构**是一种高级的Web应用程序架构方式,其核心在于通过不同的层次来实现软件系统的各个部分,从而提高...
**JSF规范详解** JavaServer Faces (JSF) 是一种用于构建Web应用程序的MVC(Model-View-Controller)框架,由Java Community Process (JCP) 的JSR-252专家组开发。JSF 1.2规范是该框架的一个重要版本,提供了组件化...
2. **生命周期流程图**:描绘了JSF请求处理的每个阶段,帮助理解何时何地执行特定的操作。 3. **事件处理图**:解释了用户交互如何转化为JSF事件,以及这些事件如何被处理。 4. **数据流图**:显示了从用户输入到...
4. **生命周期**:JSF有自己的一套请求处理生命周期,包括初始化、应用请求值、过程验证、更新模型值、调用应用逻辑和渲染响应等阶段。JSF 1.2优化了这个生命周期,提供了更多的控制点和错误处理机制。 5. **...
JavaServer Faces (JSF) 2.0 是Java EE平台中的官方组件化视图技术,专为构建Web应用程序而设计。JSF的核心特性包括预定义的...学习JSF 2.0不仅可以提升你的Web开发技能,还有助于理解企业级Java应用程序的架构和流程。
- JSF处理请求时会经历一系列的生命周期阶段,包括恢复视图、应用请求值、处理验证、更新模型值、调用应用事件和渲染响应。 - 每个阶段都有特定的任务,例如在处理验证阶段,JSF会检查用户输入是否符合预设规则。 ...
### JSF框架详解 #### 一、什么是JSF? JavaServer Faces (简称JSF) 是一个基于Java EE标准的Web应用程序框架,旨在简化Web应用程序的开发过程。JSF的目标是提供一套标准化的方法来构建和管理用户界面。它不仅提供...
**JSF(JavaServer Faces)框架入门视频资料详解** JSF,全称为JavaServer Faces,是Oracle公司开发的一种用于构建用户界面的Java EE标准组件模型框架。它为开发者提供了构建Web应用程序的简单、声明式的方法,特别...
在这些阶段中,JSF处理用户的输入、验证数据、更新模型,并调用后端业务逻辑。 3. **FacesContext**:JSF框架中的关键对象,它是与当前请求相关的上下文,包含了请求、响应、组件树、消息和配置信息。 4. **...
**JSF 2.0 实例详解** JavaServer Faces(JSF)是Java平台上用于构建Web应用程序的一种模型-视图-控制器(MVC)框架。JSF 2.0是其一个重要版本,引入了许多增强功能,提高了开发效率和用户体验。在本篇中,我们将...
- **错误处理与验证**:JSF的验证框架在源代码中清晰可见,了解其工作流程可以帮助你编写自定义验证器。 5. **扩展与自定义** - **自定义组件**:通过继承`UIComponent`或其子类,可以创建自定义组件,源代码提供...
**生命周期** 是JSF处理请求的关键步骤,包括恢复视图、应用请求值、处理验证、更新模型值、调用后处理方法和渲染响应等多个阶段。理解每个阶段的作用可以帮助开发者有效地控制和调试应用程序。 此外,**...
SWF为JSF提供了更强大的流程控制能力,而SpringFaces则是Spring社区为了更好地集成JSF而创建的项目。 **4. 整合步骤** - 配置Spring:在Web应用的`web.xml`中配置Spring的DispatcherServlet,并添加Spring的上下文...
它通过提供一套声明式、组件化的方法,使得开发者能够更加专注于应用程序的业务逻辑,而将界面布局和交互逻辑交由JSF处理。 JSF的一个核心特点是它的生命周期管理机制,该机制自动处理了诸如请求处理、渲染响应等...
6. **Faces Context**: Faces Context是JSF上下文对象,它在整个JSF请求处理过程中起着关键作用。开发者可以通过它来获取当前请求的信息,执行动作,或者访问其他JSF组件。 7. **Navigation**: JSF提供了导航机制,...
### JSF实战练习开发知识点详解 #### 一、JSF简介及重要性 **JSF**(JavaServer Faces)是一种基于组件的模型视图控制器(Model-View-Controller, MVC)框架,用于简化Java Web应用程序的开发过程。随着互联网的发展,...