`

JSP JSF tag关系

    博客分类:
  • Java
 
阅读更多
zz from http://drewdev.blogspot.com/2008/03/build-time-vs-render-time.html
另 参考 http://news.sina.com.cn/pc/2009-11-13/326/1504.html

Build time vs. render time
Overview

There always seems to be someone posting about how MyFaces or JSF is "broken" and it turns out that they misused JSTL tags. No matter how many times a solution is posted, the issue comes up again. Therefore, I am writing this blog to help those that do not understand how facelets and JSP view handlers work.
The Problem

JSF works entirely differently than JSP. JSP, when used for JSF, is used to build a JSF component tree, not render a page. JSTL tags alter the contents of what is built, the component tree, not the HTML that is output by the component renderers.
A Background of JSP

JSP was written so that authoring servlets would be easier. JSP pages are servlets, nothing more. Using a JSP compiler, a JSP file is compiled into a Java Servlet. Before tag libraries were introduced, JSP pages simply were compiled down into System.out.println(...); statements.

Tag libraries enhanced JSP by providing an API to tag authors to write Java code that would not have to be embedded in <% ... %> tags. The tags gave developers access to the text content of the tag body so that they could alter it or use it in a custom way. After some time, Sun released the Java standard tag library API, also known as the JSTL.
The JSTL

The JSTL is a set of JSP tags to perform common JSP functionality (note that they are not related to JSF in any way at all). Many of them perform logic to alter their contents. In the case of c:if and c:choose, they choose which tag contents to be included in the response or not.

I will be mainly talking about the JSTL core tags in this article (catch, choose, forEach, forTokens, if, etc.). It is these that have such a profound, and often confusing to some people, affect on JSF.
What JSF Is

JSF is a component technology, and architecturally is not related to JSP at all. In fact, using JSP with JSF has drawbacks, included serious performance implications due to the architecture of JSP and how the JSP view handler works.

    Note

    Sun probably made the default view handler as one that uses JSP, so they would not have to admit that JSP did not meet the needs of users and needed to be replaced.

Component Technology

What I mean by JSF being a component technology, is that HTML is produced by the processing of a component tree by renderers. This design is much more similar to Swing than it is Servlets and JSP. Regardless of the view handler that is used, a tree of components is built.

    Note

    A component is simply a Java Object that implements the UIComponent interface, nothing more.

These components simply store attributes, have some behaviors and are similar in nature to the Swing Tree component. JSF relies on renderers (classes that extend Renderer) to produce content from the component tree. During the restore view, a JSF view should be restored to its previous state, so that it appears to code that there was never a round trip for the code to the client (this is important as I discuss the JSTL tags and especially c:forEach).

    Note

    Components do not have to use renderers to produce their content, but it is considered bad design to use the encode methods in a component to render a response.

    Note

    JSF is typically used to produce HTML, but can be used to generate any output that doesn't even have to be XML related, although the ResponseWriter's API is designed for XML content.

JSF with JSP

JSF is implemented using JSP by default (see my note above why this stinks). What this means is that JSP is used to build the JSF component tree. This is important to note because JSP is not used to generate any of the HTML, unlike a typical JSP page. There are two main stages of JSF on JSP:

    Building of the component tree using JSP tags
    Rendering of the component tree using the standard JSF lifecycle

Building of the component tree using JSP tags

As I mentioned earlier, JSF is a component technology. When JSP tags are used, instead of writing text (HTML) onto the servlet response, the tags create components and set attribute values on them. Most of this work is done by UIComponentClassicTagBase. The class hierarchy is:

    java.lang.Object
        javax.faces.webapp.UIComponentTagBase
            javax.faces.webapp.UIComponentClassicTagBase
                javax.faces.webapp.UIComponentELTag
                javax.faces.webapp.UIComponentTag

The doStartTag method of UIComponentClassicTagBase calls findComponent, which despite its name creates the component (if needed) and calls setProperties. Therefore, the JSF component is created when the start of the JSP tag is processed. The setProperties method is responsible for setting the attributes of the component. Typically there is a one-to-one map of JSP tag attribute to JSF component attribute, so the setProperties simply copies the attributes over, doing any necessary conversion of the string value in the JSP attribute to an object that the component expects for the attribute.
How JSTL fits in

Most of the "work" of a component takes place during rendering. For example, the h:dataTable sets up the var attribute during rendering (and other phases too, but that is not pertinent to this article). This means that EL expressions that rely on variables that only have scope during the rendering phase are not valid during component creation. That is to say, there is no component lifecycle method for when the component is created.

Since JSTL tags are plain JSP tags and do not involve the JSF lifecycle, they do not have access to the environment that components create, like the var variable of an h:dataTable. Take this code for example:

    <f:view> <h:dataTable var="_row" value="#{bean.rows}> <h:column> <c:choose> <c:when test="#{_row.editable}"> <h:inputText value="#{_row.value}" /> </c:when> <c:otherwise> <h:outputText value="#{_row.value}" /> </c:otherwise> </c:choose> </h:column> </h:dataTable> </f:view>

Although from the pure XML standpoint this looks valid, it is not. Thinking about what I just said it is a simple problem:

    Data table component created from the data table JSP tag
    Column component created from its JSP tag
    The when tag in the choose attempts to evaluate #{_row.editable}
        Since _row is not bound (yet) in the EL resolver, null is returned
        (an argument could be made that the EL engine should throw an error or at least give a warning, but that is not how it is designed)
        Since null is technically false, the outputText is created instead of the inputText
    The output text component is created by its tag

So the page author was probably expecting that the choose would be evaluated for every row in the table. But the table has no rows at this point, the component is being created. So, the inputText component is never created as the when tag will not process its contents if the test fails.

The proper solution is to use JSF functionality and not JSP functionality. To the confusion of JSP developers, there are no c:if, c:choose or c:forEach equivalent components (see the Tomahawk Sandbox limitRendered for c:if and c:choose functionality and the Tomahawk dataList or Trinidad iterator components for c:forEach type of functionality). Without third party components, the rendered attribute can also work, although it requires more work:

    <f:view> <h:dataTable var="_row" value="#{bean.rows}> <h:column> <h:inputText value="#{_row.value}" rendered="#{_row.editable}" /> <h:outputText value="#{_row.value}" rendered="#{not _row.editable}" /> </h:column> </h:dataTable> </f:view>

In this case both the input and the output components are created. The rendered is evaluated by the renderer during the rendering phase while the data table is iterating over each of its rows.
When it is okay to use JSP tags with JSF

So now that you know the problem, you may ask why is using any non-JSF, JSP tags allowed? This is because they still work, but must be used correctly. Since JSP tags are evaluated while components are being built, they can control which components to create. For example, I may want to include certain components in a page template if the current user is an administrator. Because the components are not created if the user is not an administrator, there will be less components objects created and less components to process for each JSF lifecycle phase, and thus improving performance.

It is very important to remember that you cannot have components "re-appear" on post back of a JSF form. This is because a JSF component tree should never be altered between having its state saved and when its state is restored. This is very important, so let me say again, a JSF component tree should never be altered between having its state saved and when its state is restored. The reason is that this would violate the JSF specification/design. Take for example a c:forEach loop that when evaluated had five items it its list. Assuming that there was one JSF component tag inside the for each tag, that means five components were created. Now JSF saves the state for five components. Then lets say this for each loop data is coming from a database that can change and now there are only four items. When JSF attempts to restore the component tree, there is data for 5 components, but there are only 4 in the component tree. This causes a restore state exception.

Therefore, always ensure that when a component tree is restored, it is not changed! There is no problem changing a component tree after restoring it or before saving it, only between those times.
Hey, what about Facelets?

I did mention that I would talk about facelets, didn't I?

Facelets is similar in its behavior to JSP although it has much more efficient code and is designed quite differently than JSP. Instead of JSP tags, facelets has TagHandlers. These tag handlers analyze the XML content of a facelet and create components. Just like JSP tags, the tag handlers have no access to the render-time scope of the components. In fact, facelets provides tag handlers to mimic the JSTL functionality. These tag handlers have the same pitfalls as the JSP tags. Therefore, the same considerations and rules apply to tag handlers that apply to JSP tags.
Other Blogs

Here are some other helpful resources on this topic:

    c:foreach vs ui:repeat in facelets
分享到:
评论

相关推荐

    jsp jsf beginning study source code

    4. **EL(Expression Language)和JSTL(JavaServer Pages Standard Tag Library)**:EL简化了获取和设置JavaBean属性,而JSTL提供了一系列标准标签,使得JSP页面更易于维护和阅读。 **JSF技术详解:** 1. **JSF...

    Beginning JSP JSF and Tomcat Web Development From Novice to Professional.pdf

    JSP的核心概念包括JSP元素(指令、脚本元素和动作)、生命周期(翻译、初始化、服务、销毁)、EL(Expression Language)以及JSTL(JavaServer Pages Standard Tag Library)。 2. **JavaServer Faces (JSF)**: JSF...

    JSF Java Server Faces (JSF)框架

    JSF是一种用于构建Java ... 包含JSP页面的JSF应用程序也使用由为了表现UI组件和在页面上的其他对象的JSF技术而定义的标准的tag库。 Java Server Faces技术的重要开发框架 sun-ri、myfaces、icefaces、richfaces、seam

    jsf-api.jar和jsf-impl.jar

    **JSTL(JavaServer Pages Standard Tag Library)** 是一套标准的JSP标签库,用于简化JSP页面的编写,尤其是处理通用任务如迭代、条件语句、XML处理等。JSTL分为API和实现两部分。 **jstl-api-1.2.jar** 包含了...

    开发JSF所需要的jar包

    这些JAR提供了Servlet和JSP的相关接口,是JSF运行的基础。 5. **Facelets JAR**: Facelets是JSF默认的视图技术,它是一种XML/XHTML基的视图定义语言。`com.sun.faces.facelets.jar`包含了Facelets的实现,用于...

    jsf相关jar包 jsf-api.jar jsf-impl.jar

    这个JAR文件包含Java Standard Tag Library(JSTL)1.2版本,它是一个用于JSP页面的标准标签库,提供了处理SQL查询、XML处理、国际化、条件逻辑等功能。JSTL与JSF结合使用,可以增强JSF应用的表达能力,使得开发者...

    jsf api

    `jsf12-tlddoc`可能是指JSF 1.2版本的Tag Library Descriptors (TLD) 文档,它包含了JSF 1.2标签库的详细信息,包括每个标签的功能、属性和事件处理。 **10. 实战应用** 通过学习JSF API,你可以构建出功能丰富的、...

    jstl jsf包合集

    【标题】"jstl jsf包合集"所涵盖的知识点主要集中在Java服务器端技术领域,特别是JSTL(JavaServer Pages Standard Tag Library)和JSF(JavaServer Faces)两个核心组件。这两个技术都是Java EE平台的重要部分,...

    javaee jsf-api

    开发者可以利用JSF API来设计用户界面,JSF IMPL来处理应用逻辑,JSTL来增强JSP页面的功能,而`javaee.jar`则提供了整个JavaEE环境的基础设施。了解并熟练掌握这些技术,对于开发高效、可维护的企业级应用至关重要。

    JSF配置以及简单的登陆验证

    接着,为了支持JSTL(JavaServer Pages Standard Tag Library),需要将jstl.jar文件添加到项目的WEB-INF/lib目录下。JSTL提供了一系列标签,简化了JSP页面的编写。 创建index.jsp作为登录页面,利用JSF的HTML模板...

    jsf标签官方帮助文档

    **JSF标签** 是JSF中的关键元素,它们是UI组件的表示,可以在JSP(JavaServer Pages)或其他支持JSF的视图技术中使用。这些标签定义了在用户界面上显示的内容和行为。例如,`&lt;h:outputText&gt;` 标签用于显示文本,而 `...

    jsf学习必备——实用工具书

    压缩包中的"tlddoc"可能是指Tag Library Documentation,这是对JSF中使用的标签库的文档。TLD文件描述了库中每个标签的用途、属性、事件等信息,对于理解和使用特定库非常有用。例如,可能包含了关于PrimeFaces、...

    jsf-api,jsf-impl,jst1-1.2,javaee

    "jstl-1.2"指的是JSTL(JavaServer Pages Standard Tag Library),这是一个用于JSP页面的标准标签库,它包含了一系列预定义的标签,可以用来处理常见任务,如迭代、条件判断、XML处理、国际化和URL重写等。...

    JSF开发所需的jar包

    3. **jstl-1.2.jar**:JSTL(JavaServer Pages Standard Tag Library)是JSP的标准标签库,提供了处理常见任务如循环、条件判断、国际化等功能的标签。在JSF中,JSTL可以与EL(Expression Language)配合使用,增强...

    javaee.jar,jsf-api.jar,jsf-impl.jar,jstl-1.2.jar

    4. **jstl-1.2.jar**:JSTL(JavaServer Pages Standard Tag Library)是一组预定义的标签库,用于简化JSP页面的开发。JSTL 1.2版本包括了核心标签、JDBC标签、XML标签和国际化的支持。这些标签可以用来处理常见任务...

    JSF下载 包含所需的七个jar包

    3. **standard.jar**: 这通常包含Servlet容器中的一些标准JSP标签库,如JSTL(JavaServer Pages Standard Tag Library)的基础支持。这些标签库帮助开发者更方便地编写JSP页面,减少脚本代码的使用。 4. **jsf-api....

    JSF2.2基本jar包

    JSTL(JavaServer Pages Standard Tag Library)是JSP标准标签库,提供了处理XML、SQL、国际化、逻辑控制等功能的标签。在JSF项目中,`jstl.jar`常常被用来增强视图层的表现力,比如使用 `&lt;c:if&gt;`、`&lt;c:forEach&gt;` ...

    一个简单的jsf例子------JSF2学习笔记1

    - `jstl.jar` 包含JSTL (JavaServer Pages Standard Tag Library),提供了用于JSP页面的标签库,如条件语句、迭代和函数支持。 这些库文件是JSF2运行环境的组成部分,它们提供了JSF所需的类和功能,帮助开发者构建...

    JSF标签暨注解规范

    5. **JSTL核心库**和**JSTL函数库**:这两个库提供了JSP Standard Tag Library的标签,如`&lt;c:if&gt;`和`&lt;c:forEach&gt;`,以及函数库如`&lt;fn:length&gt;`,增强了JSF应用的功能。 6. **Facelets Templating库**:提供模板和...

    JSF web jar包

    6. **依赖的库**:JSF可能会依赖于其他库,比如CDI(Contexts and Dependency Injection)、JSTL(JavaServer Pages Standard Tag Library)等,这些都需要被正确引入。 为了将这些jar包引入到项目中,开发者通常会...

Global site tag (gtag.js) - Google Analytics