读书人

JSF1.x引见及其弊端

发布时间: 2012-10-27 10:42:26 作者: rapoo

JSF1.x介绍及其弊端

?

先来看看JSF的组成:

JSF1.x引见及其弊端

Template (jsp) 即指用户界面;Faces-config 定义导航规则和后台Bean;Backing beans 处理action, 导航处理,作为业务逻辑层的连接者;Wrapper bean 封装数据POJOS对象,包含了一些UI对象的属性;Logic layer beans 业务逻辑Bean,Model 数据POJO对象模型

?

Template的例子:

<%@ taglib uri="http://java.sun.com/jsf/core" prefix="f"%><%@ taglib uri="http://java.sun.com/jsf/html" prefix="h"%><f:view><html><head><title>Items</title></head><body><h:form id="items">   <h:dataTable id="itemlist” value="#{JsfBean.allItems}” var="entry">      <h:column>         <f:facet name="header">            <h:outputText value=""/>         </f:facet>         <h:selectBooleanCheckbox id="itemSelect" value="#{entry.selected}" rendered="#{entry.canDelete}"/>         <h:outputText value="" rendered="#{not entry.canDelete}"/>      </h:column></h:form></body></html></f:view>
    大部分情况用JSP作为Template严重依赖于标签库taglibsCore(f)-基本页面组件标签Html(h)-html标准标签库写一个新的标签是件比较困难的事情大量使用表达式语言 EL (expression language)

?

faces-config.xml的例子

<faces-config>   <navigation-rule>    <from-view-id>/jsf/JsfItems.jsp</from-view-id>    <navigation-case>        <from-outcome>newItem</from-outcome>        <to-view-id>/jsf/JsfAddItem.jsp</to-view-id>    </navigation-case>    <navigation-case>        <from-outcome>*</from-outcome>        <to-view-id>/jsf/JsfItems.jsp</to-view-id>    </navigation-case>   </navigation-rule>   <managed-bean>        <managed-bean-name>JsfBean</managed-bean-name>        <managed-bean-class>org.example.jsf.JsfBean</managed-bean-class>        <managed-bean-scope>session</managed-bean-scope>    <managed-property>        <property-name>logic</property-name>        <value>#{someLogicBean}</value>    </managed-property>   </managed-bean></faces-config>
?定义导航规则,这里定义了在JsfItems.jsp上所有的对外连接或者按钮对应的下一个视图的名称from-outcome和视图路径to-view-id,这两个单词真不专业,很难理解。以下两个按钮,点击后都会跑到"newItem"所对应的<to-view-id>/jsf/JsfAddItem.jsp</to-view-id>上:
<h:commandButton id="submit" action="newItem" value="Submit" /> <h:commandButton id="submit" action="#{JsfAppBean.processActionNew}" value="Submit" /> 
?其中,JsfAppBean.processActionNew返回"newItem"
public String processActionNew() {   currentItem = null;   itemText = TEXT_DEFAULT;   itemHidden = HIDDEN_DEFAULT;   return "newItem";}
?
对于*匹配到out-outcome,表示所有其他链接都链到其对应的to-view-id上。受托管bean-MBean:managed-bean-name是这一MBean的名字,用于其它位置的调用;例如:
<h:dataTable id="itemlist” value="#{JsfBean.allItems}” var="entry">...
FacesContext context = FacesContext.getCurrentInstance();ValueBinding binding = context.getApplication().createValueBinding("#{JsfBean}");UserBean user = (UserBean) binding.getValue(context);
?
managed-bean-class是这一MBean的完整路径,用于指明该MBean的类文件位置;managed-bean-scope是这一MBean的有效范围;application, request, session,nonemanaged-property是MBean的托管属性,例如此处JSF的IOC容器JSF Managed Bean Facility(MBF)会将初始化好的someLogicBean自动通过JsfBean的setLogic方法注入logic属性。(someLogicBean的配置被忽略了,也是指一个受托管的Bean)。

?

?

接下来看一个后台Bean的例子:

public class JsfBean {    private DataModel itemsModel;    private JsfItemWrapper currentItem = null;...    private JsfLogic logic;    public void setLogic(JsfLogic logic) {       this.logic = logic;    }...    public DataModel getAllItems() {       List wrappedItems = new ArrayList();       List items = logic.getAllVisibleItems(logic.getCurrentSiteId());       for (Iterator iter = items.iterator(); iter.hasNext(); ) {        JsfItemWrapper wrapper =            new JsfItemWrapper((Item) iter.next());           wrappedItems.add(wrapper);       }       itemsModel = new ListDataModel(wrappedItems);       return itemsModel;    }...    public String processActionList() {       return "listItems";    }}
?包含了一个wrapped bean“JsfItemWrapper”;所有get和set用来存储和取出界面层需要的数据;方法用于响应按钮的事件,并返回一个视图字符串,对应上面配置中的from-outcome;业务逻辑层对象logic通过托管属性方式JSF IOC自动注入;
<managed-bean>        <managed-bean-name>JsfBean</managed-bean-name>        <managed-bean-class>org.example.jsf.JsfBean</managed-bean-class>        <managed-bean-scope>session</managed-bean-scope>        <managed-property>            <property-name>logic</property-name>            <value>#{someLogicBean}</value>        </managed-property></managed-bean>
?

Wrapped Bean主要是被JSF引用的数据POJO封装对象

public class JsfItemWrapper {private Item item;private boolean isSelected; // is this item selected by the userpublic JsfItemWrapper(Item item) {this.item = item;}public Item getItem() {return item;}public void setItem(Item item) {this.item = item;}public boolean isSelected() {return isSelected;}public void setSelected(boolean isSelected) {this.isSelected = isSelected;}}
封装此类对象的目的,主要是避免将UI的信息放在data对象中。JSF等于在java端更多体现了UI组件和其控制。

通过快速学习JSF,可以看出,JSF1.x开发存在以下问题:

要学习很多UI标签库组件,不过struts也一样,建议项目组将要用的组件列出参考手册,也好办;很难进行Ajax编程,在2.0彻底改善了;没有回退,没有Get,都是POST,所以,无法保存URL?,在2.0彻底改善了;维护faces-config.xml挺麻烦的;UI设计人员会比较麻烦,不过,纵观所有以标签库为基础的页面,都存在这个问题,现在很多开发工具都支持JSF可视化开发了,但好像不是针对UI设计人员;由于JSF将在后台可以直接访问前台UI组件的属性和方法,这样导致一般情况需提供一个Wrapped Bean了。

读书人网 >JavaScript

热点推荐