javaserver pages ( jsp)和xml是sun的j2ee的两个至关紧要的组件。 jsp是用于创建应用程序服务器端程序的有效工具,而客户可以是一个浏览器,一个设备或其它的应用程序。 你可以使用xml描述数据并在联系服务器与系统的其余部分之间传递。 如果你仔细考虑web服务的抽象概念的话,jsp可以被认为是实现技术而xml则是数据封装和消息传送技术。 jsp页面可以通过三种方式使用xml:直接使用xml文件,使用javabeans来执行xml处理或者通过标记库使用xml。
一、直接使用xml
我们可以在jsp页面中直接使用xml,这要分为三类:
1. jsp可以读取xml文件并且基于这些数据执行动作。例如:一应用程序可以读取具有某些特定结构的数据的xml文件。
2. jsp可以创建xml文件以发送数据到客户程序或其它的应用程序。 jsp可以转换xml文件,这个变换可以是交给xslt处理,由jsp作为控制器,或者通过非xslt解决方案来完成。 在这两种情况下,jsp的作用都是读取xml文件,转换它并生成一个输出。
因为jsp包含内嵌的java程序,所以它可以直接地调用一个分析程序来读/写xml数据。 这是非常不合理的方法,因为数据和代码逻辑并不能很好的被分离开。 另外,这样的程序也很难读得懂。 所以,下面我将介绍第二方法:使用javabeans。
二、使用javabeans
jsp可以通过<jsp:usebean>标记与javabeans紧密的整合起来。 下面的程序片段示范了如何使用在jsp页面中的一个javabean来设置并获取属性。
<jsp:usebean id="cb" scope="session" class="xmlrep.customer" />
<jsp:setproperty name="cb" property="id" value="45" />
<b> first name is: </b>
<%=cb.getfname() %>
<p>
<b> last name is: .</b>
<%= cb.getlname() %>
jsp与javabeans整合的特征就是可以自动把超文本标志语言的表单元素翻译成javabean属性。 如果有一个html表单并且想让它向javabean提交表单内容,你可以写下面的代码:
<jsp:setproperty name="cb" property="*" />
name属性包含jsp页面已经引用bean的值。 前面<jsp:usebean>标记设置名称为" cb "。 与设置单独的bean属性不同,你可以使用星号来标志"全部的"属性。 jsp页面自动地映射html表单值为同名的bean属性。 如果你读取每个html表单元素,然后调用相应属性的bean设置方法,那么结果也将是相同的。
可以看到,类xml的标记允许jsp页面访问javabeans。 我们通过尽可能多的把封装的代码转变成可重用组件(javabeans),就可以把jsp页面中代码优化到最小的程度。 你可以使用通用的语法分析程序,像xerces或者jaxpi,在单独的javabeans里与xml文件交互作用--并且你可以在不改动jsp页面的情况下改变分析程序。 此外,beans还可以使用xslt来执行xml文件的转换。
使用jsp和javabeans来完成的这些抽象的动作比直接在jsp页面中插入原始的java程序要好得多,但是你仍然需要熟悉java程序,以便随时改变jsp页面。 应用程序的一致性与条理性依靠javabeans合作创建一个统一的输出结果的好坏程度。 例如,bean中的缺陷可能会造成整个xml输出无效。 依靠beans指定资源的方法,可能同时带来执行性能的问题。
三、通过标记库jsp与xml交互
这也就是在前文中重点提到的,但是因为它实在太重要了,所以在本文中我必须也要提到。标记库可以定义出现在jsp页面中的作为类xml元素的自定义标签,可以把特定的java代码与每个标记都关联起来。 例如,假定你可以访问一个天气情况数据库,而且你需要输出现在的天气状况。 那么,你可以在jsp中插入jdbc程序代码直接查询数据库(虽然这并不是一个好的选择),把这些代码封装到一个javabean内,或者把它包装成一个标记库。 使用最后一个选择,你的jsp页面内的程序代码看上去就像:
<%@taglib uri="the tld file" prefix="foo" %>
current weather is <foo:weather/>
注意在上述程序代码中看不到任何java代码的踪迹。 作为一个页面设计人员,你使用一种熟悉的语法就像<foo:weather/>,这些与其它的任何标记看起来非常相似。 在页面中包含现在的天气状况的html字符串的地方插入它。
标记库有一个关联的xml格式的描述符文件,名叫tag library descriptor(标记库描述符,tld)。 在tld文件中的每个标记都有一个条目,包括它的名称,版本和其它的任选信息。 在jsp页面内,你可以用" <%@_taglib prefix = " foo " %> "指令指定tld文件。 属性" prefix "用来指定一个前缀,用来在jsp页面内使用特定的库来引用任何标记。 那为什么我们要使用标记<foo:weather/>而不仅仅是<weather/>。 tld文件的精确位置依赖于正在使用的应用程序服务器。
一个标记库标记可以代替为了完成这段程序逻辑的相应的java程序代码。 每个标记都相当于一个相同名称的java类。 这个类必须实现tagsupport接口,包含捕获事件触发器方法作为处理这个页面的jsp引擎。 当它第一次遇到这个标记的时候,引擎就会调用dostarttag ()方法。 可以使这个方法为空或者当需要的时候才执行应用程序逻辑。 当该方法返回skip_body时,那么引擎跳过这个标记体。 当它返回eval_body_include时,引擎将会处理这个标记以及它的子标记。 类似地,jsp引擎在分析了结束标记之后就会调用doendtag ()方法。 doafterbody ()方法让你可以在引擎处理元素体之后执行动作,但是必须在doendtag ()方法作用之前。 下面就是实现天气状况的weather类的一个样本程序代码:
import javax.servlet.jsp.*;
import javax.servlet.jsp.tagext.*;
import java.io.*;
public class weather extends tagsupport {
public int dostarttag() {
try {
jspwriter out = pagecontext.getout();
out.print(" sunny and cloudy mixed with " +
"rain and sunshine. ");
} catch (ioexception e) {
system.out.println("error " + e);
}
return (skip_body);
}
}
当引擎遇到" <someprefix:weather/> "标记的时候,它会在标记库内搜索一个同名的类。 如果dostarttag ()方法被实现的话(在本例中是这样的),它就会被调用。 这就使字符串包含适应显示的天气情况。 因为方法返回了skip_body,jsp读取器移动到标记的末尾。
最简单的使用jsp和标记库的方法就是使用apache tomcat引擎。 这个引擎也充当servlet和jsp应用程序接口的引用实现。
当使用标记库时,jsp页面看上去就非常像xml文件了。 当jsp页面被处理时,引擎执行与标记相关联的程序代码(实际上,首先调用jsp引擎把jsp页面翻译成一个servlet,然后再编译servlet。 与标记库相关联的方法都被包含在servlet中。),一个熟悉xml的人就可以设计并使用各种各样的页面布局进行试验,而不必改变任何java程序代码。当然,代码与数据的分离的程度还是主要依赖标记库元素设计的好坏程度。
新闻热点
疑难解答