Struts和JSF/Tapestry都属于表现层框架这两种分属不同性质的框架后者是一种事件驱动型的组件模型而Struts只是单纯的MVC模式框架老外总是急吼吼说事件驱动型就比MVC模式框架好何以见得我们下面进行详细分析比较一下到底是怎么回事?
首先事件是指从客户端页面(浏览器)由用户操作触发的事件Struts使用Action来接受浏览器表单提交的事件这里使用了Command模式每个继承Action的子类都必须实现一个方法execute
在Struts中实际是一个表单Form对应一个Action类(或DispatchAction)换一句话说在Struts中实际是一个表单只能对应一个事件Struts这种事件方式称为application eventapplication event和Component event相比是一种粗粒度的事件
Struts重要的表单对象ActionForm是一种对象它代表了一种应用这个对象中至少包含几个字段这些字段是Jsp页面表单中的input字段因为一个表单对应一个事件所以当我们需要将事件粒度细化到表单中这些字段时也就是说一个字段对应一个事件时单纯使用Struts就不太可能当然通过结合JavaScript也是可以转弯实现的
而这种情况使用JSF就可以方便实现
<h:inputText id=userId value=#{loginuserId}>
<f:valueChangeListener type=logindemoUserLoginChanged />
</h:inputText>
#{loginuserId}表示从名为login的JavaBean的getUserId获得的结果这个功能使用struts也可以实现name=login property=userId
关键是第二行这里表示如果userId的值改变并且确定提交后将触发调用类UserLoginChanged的processValueChanged()方法
JSF可以为组件提供两种事件Value Changed和 Action 前者我们已经在上节见识过用处后者就相当于struts中表单提交Action机制它的JSF写法如下
<h:commandButton id=login commandName=login>
<f:actionListener type=logindemoLoginActionListener />
</h:commandButton>
从代码可以看出这两种事件是通过Listerner这样观察者模式贴在具体组件字段上的而Struts此类事件是原始的一种表单提交Submit触发机制如果说前者比较语言化(编程语言习惯做法类似Swing编程)后者是属于WEB化因为它是来自Html表单如果你起步是从Perl/PHP开始反而容易接受Struts这种风格
基本配置
Struts和JSF都是一种框架JSF必须需要两种包JSF核心包JSTL包(标签库)此外JSF还将使用到Apache项目的一些commons包这些Apache包只要部署在你的服务器中既可
JSF包下载地址选择其中Reference Implementation
JSTL包下载在 /downloads_taglibsstandardcgi
所以从JSF的驱动包组成看其开源基因也占据很大的比重JSF是一个SUN伙伴们工业标准和开源之间的一个混血儿
上述两个地址下载的jar合并在一起就是JSF所需要的全部驱动包了与Struts的驱动包一样这些驱动包必须位于Web项目的WEBINF/lib和Struts一样的是也必须在webxml中有如下配置
<webapp>
<servlet>
<servletname>Faces Servlet</servletname>
<servletclass>javaxfaceswebappFacesServlet</servletclass>
<loadonstartup></loadonstartup>
</servlet>
<servletmapping>
<servletname>Faces Servlet</servletname>
<urlpattern>*faces</urlpattern>
</servletmapping>
</webapp>
这里和Struts的webxml配置何其相似简直一模一样
正如Struts的strutsconfigxml一样JSF也有类似的facesconfigxml配置文件
<facesconfig>
<navigationrule>
<fromviewid>/indexjsp</fromviewid>
<navigationcase>
<fromoutcome>login</fromoutcome>
<toviewid>/welcomejsp</toviewid>
</navigationcase>
</navigationrule>
<managedbean>
<managedbeanname>user</managedbeanname>
<managedbeanclass>rejsfUserBean</managedbeanclass>
<managedbeanscope>session</managedbeanscope>
</managedbean>
</facesconfig>
在Strutsconfigxml中有ActionForm Action以及Jsp之间的流程关系在facesconfigxml中也有这样的流程我们具体解释一下Navigation
在indexjsp中有一个事件
<h:commandButton label=Login action=login />
Action的值必须匹配formoutcome值上述Navigation配置表示如果在indexjsp中有一个login事件那么事件触发后下一个页面将是welcomejsp
JSF有一个独立的事件发生和页面导航的流程安排这个思路比struts要非常清晰
managedbean类似Struts的ActionForm正如可以在strutsconfigxml中定义ActionForm的scope一样这里也定义了managedbean的scope为session
但是如果你只以为JSF的managedbean就这点功能就错了JSF融入了新的Ioc模式/依赖性注射等技术
Ioc模式
对于Userbean这样一个managedbean其代码如下
public class UserBean {
private String name;
private String password;
// PROPERTY: name
public String getName() { return name; }
public void setName(String newValue) { name = newValue; }
// PROPERTY: password
public String getPassword() { return password; }
public void setPassword(String newValue) { password = newValue; }
}
<managedbean>
<managedbeanname>user</managedbeanname>
<managedbeanclass>rejsfUserBean</managedbeanclass>
<managedbeanscope>session</managedbeanscope>
<managedproperty>
<propertyname>name</propertyname>
<value>me</value>
</managedproperty>
<managedproperty>
<propertyname>password</propertyname>
<value>secret</value>
</managedproperty>
</managedbean>
facesconfigxml这段配置其实是将me赋值给name将secret赋值给password这是采取Ioc模式中的Setter注射方式
Backing Beans
对于一个web form我们可以使用一个bean包含其涉及的所有组件这个bean就称为Backing Bean Backing Bean的优点是一个单个类可以封装相关一系列功能的数据和逻辑
说白了就是一个Javabean里包含其他Javabean互相调用属于Facade模式或Adapter模式
对于一个Backing Beans来说其中包含了几个managedbeanmanagedbean一定是有scope的那么这其中的几个managedbeans如何配置它们的scope呢?
<managedbean>
<managedproperty>
<propertyname>visit</propertyname>
<value>#{sessionScopevisit}</value>
</managedproperty>
这里配置了一个Backing Beans中有一个setVisit方法将这个visit赋值为session中的visit这样以后在程序中我们只管访问visit对象从中获取我们希望的数据(如用户登陆注册信息)而visit是保存在session还是application或request只需要配置既可
UI界面
JSF和Struts一样除了JavaBeans类之外还有页面表现元素都是是使用标签完成的Struts也提供了strutsfacestld标签库向JSF过渡
使用Struts标签库编程复杂页面时一个最大问题是会大量使用logic标签这个logic如同if语句一旦写起来搞的JSP页面象俄罗斯方块一样但是使用JSF标签就简洁优美
<jia:navigatorItem name=inbox label=InBox
icon=/images/inboxgif
action=inbox
disabled=#{!authenticationBeaninboxAuthorized}/>
如果authenticationBean中inboxAuthorized返回是假那么这一行标签就不用显示多干净利索!
先写到这里我会继续对JSF深入比较下去如果研究过Jdon框架的人可能会发现Jdon框架的jdonframeworkxml中service配置和managedbean一样都使用了依赖注射看来对Javabean的依赖注射已经迅速地成为一种新技术象征如果你还不了解Ioc模式赶紧补课