JSF1.x介绍及其弊端
?
先来看看JSF的组成:

?
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>?
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";}?<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);??
?
接下来看一个后台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了。