JSP设计模式基础:View Helper模式
——学习如何使用View Helper模式使得Model数据适应表现层的需要
(第一部分)
作者:Andrew Patzer
前言
这篇从《JSP设计模式基础》(Apress, 2004)摘录的文章,描述了View Helper模式,并且向我们展示了如何创建一些能增加到你自己的工具箱的有用的View Helper(2,300 字, November 1, 2004) 。
View Helper模式告诉我们,我们能使用Helpers来使得在一个应用中,Model数据能够适应表现层的需要。典型的,表现层一般都包括一些JSP页面。这些页面由一些用来给用户显示内容的HTML和图片组成。然而,当这些页面需要显示一些存储在Model上的动态信息的时候,这里有一个问题出现了。你希望能够避免在页面上为了显示那些动态数据而使用嵌入的Java代码,你就得使用一些Helper来帮助你实现上述的功能。
要抛弃那些在控制器Servlet里将一些Model数据作为一个属性存储在request里面的想法。在一个页面里,你要获取Model数据,有三种选择。你可以以JSP脚本的形式嵌入Java代码;你也可以使用EL;或者你也可以使用一个Helper帮你取得数据。根据将表现层和商业逻辑分离的原则,使用一些Helper来帮助我们使得数据适应表现层的要求比将表现层代码和Java代码混在一起有意义(参见图1)。
图1
你可以设想,通过使用简单易用的Helper代替Java代码,页面设计人员开发表现层变得简单多了。当然,前提是开发人员发布了一个Helper目录并且详细描述了这些Helper怎么使用;因为这样开发人员才会使用那些Helper。然而,如果在开发人员能够提供使Helper运行起来的Model数据之前,页面设计人员就已经设计了页面,就又有新的问题。解决这个问题的一个有用的技巧是在Helper里设置一些假数据以便在没有Model数据的时候显示;还有一个可以替代的方法是在Model里设置一些能使Helper运行的假数据。无论哪种方法,页面设计人员都不会在等待开发人员的时候闲起来。
使用Helper有如下优点:
。表现层的组件是标准化的,为应用提供了统一的look和feel
。Java代码从页面设计人员那里被抽离,使得他们有了易用的Helper来访问Model
。如果Model还不存在的话,你可以创建一些Helper来显示一些假数据。这样,不管应用程序的准备如何,页面设计人员都可以进行他们的设计工作。
。Helper作为业务数据和表现层的中介,它可以把两者清晰的分离开来。
实现JavaBeans Helper
的策略
在为JSP页面开发Helper的时候,你有两种选择。你可以使用JavaBeans或者定制标签,具体选择哪一种取决于你在Helper中所要处理的数据。一般来说,JavaBeans适合你处理单个的数据,而定制标签却更适合使用在那些处理一系列数据的场合。然而,需要着重指出的是,你可以用任意一种方法来处理两种类型的数据。
实现View Helper模式的策略
你可以在一个JSP页面里用JavaBeans来实现Helper。当处理和格式化单一的文本数据时,JavaBeans模式的Helper非常简单易用。那些内置的JSP标签会让你非常简单和直观的使用JavaBeans。对JavaBeans的使用包括简单的声明,后面就可以引用该给定的标签了,就像下面那样:
<%-- Declare bean --%>
<jsp:useBean id="myBean" class="jspBook.util.myBean"/>
<%-- Get first name from bean --%>
Hello <jsp:getProperty name="myBean" property="firstName"/>,
welcome to Acme Products' online store!
JavaBeans能做的事可不仅仅是简单的将数据项从Model里取出来,它还能格式化制定的数据项、进行计算或产生大块的数据项。如你所想,它非常适合使用内嵌的JSP标签来获取数据项。但是如果你使用JavaBeans太多,那么你的JSP页面将因为太多的Java代码而变得混乱不堪,不管你使用多少EL。在这种情况下,你应该把所有的附加行为封装到一个制定标签里面去。
实现定制标签Helper
的策略
为了对付复杂Model的转换,定制标签能够嵌入Java代码,操作好几个有关数据的算子,只提供简单的标签给页面设计者使用。为了使用定制标签,你必须写一个继承了TagSupport或BodyTagSupport的类。你可以在标签库描述符里声明你的类,如下所示:
Listing 1. An Example TLD
<?xml version="1.0" encoding="ISO-8859-1" ?>
<taglib xmlns="http://java.sun.com/xml/ns/j2ee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee
web-jsptaglibrary_2_0.xsd"
version="2.0" >
<tlib-version>1.0</tlib-version>
<jsp-version>2.0</jsp-version>
<short-name>myTags</short-name>
<description>
Tag library to support the examples in Chapter 8
</description>
<tag>
<name>myTag</name>
<tag-class>jspbook.ch08.myTag</tag-class>
<body-content>JSP</body-content>
<attribute>
<name>myAttribute</name>
<required>yes</required>
</attribute>
</tag>
</taglib>
通过在JSP页面里首先使用taglib指示符声明以后,这个定制标签就能够在页面里被引用,如下所示:
<%@ taglib uri="/helpers" prefix="helpers" %>
<helpers:myTag myAttribute="some value">
Body text...
</helpers:myTag>
我更倾向于使用定制标签作为View Helper的实现方式。因为当它们集中存储在应用服务器的时候,它们给了开发人员更多的访问Servlet上下文的权限并且提供了更多的性能优点。另一个我倾向于使用定制标签的原因是它更利于非Java开发人员直观的使用,它们的格式更像标准的HTML标签,这些HMTL标签对于我们大多数的人来说再熟悉不过。最后,一旦这些定制标签经过了你的开发和测试,你就可以在你的整个项目的所有JSP页面使用它们。一旦这些定制标签被设计得更加通用,那么你可以将它们打包起来在所有的项目中使用到它们。
实现Model
分离的策略
不管是使用定制标签或者JavaBeans,提供独立的Helper是非常有用的,这些独立的Helper能够在没有Model数据存在的时候提供一系列的假数据来代替Model数据。这使得页面开发人员可以独立于开发团队而完成它们的任务。为了实现这种策略,Helper需要去检测Model的存在,以便使用一个真实的Model数据或者使用一个静态的假Model数据(如下图)。
按照这个思路,我们需要创建一个静态的Model来精确复制真实的Model。不是所有的时候都能让这两者保持同步。一个在某些时候更完美的替代方法是,让开发人员创造一些假数据到Model里头,以便页面开发人员能够当作真实Model已经存在一样;同时也确保了他们自己工作的那个Model永远是正确的(见图三)。