在Jsp中模拟WebForm(二)
用“页面处理器”的方式组织页面代码,起到了将页面显示元素与服务器端控制代码分离的目的,使得我们的代码更为清晰。在页面上,需要例行公事地调用页面相应的处理器(多个页面可以具有相同的处理器),声明一个类似on_event的javaScript函数,并在需要进行“回调”(提交本页面,并重新请求本页面)的Form控件的事件中调用on_event(同时指定事件的名称和参数),还需要指定Form的Action指向本页面,并在form中放置两个隐藏字段,分别持有页面发生的事件名称和需要向服务器传递的参数。
是的,需要例行公事作这么多事情,拷贝和粘贴可以完成这些工作,但如果自定义一个标签,则可以将这些类似要做的工作自动完成。我们先定义一个标签(PageTag),向其指定页面的处理器类名,由其负责调用页面处理器:
public class PageTag extends BodyTagSupport
{
protected String pageHandlerClass = null;
PageHandler handler = null;
final public String getPageHandler()
{
return (this.pageHandlerClass);
}
//为标签定义PageHandler属性
final public void setPageHandler(String pageHandler)
{
this.pageHandlerClass = pageHandler;
}
//当标签开始执行时,调用页面处理器
final public int doStartTag() throws JspException
{
//生成PageHandler的实例
try
{
Class p = Class.forName(pageHandler);
//在这里我们需要修改一下前面定义
//的PageHandler类,为其定义无参的构造函数,取消先前的构造
handler = (PageHandler) p.newInstance();
//改变process的定义,为其传入一个PageContext,并实现由原先
//的构造所完成的功能(取得操作页面的基本组件)。
handler.process(this.pageContext);//执行页面处理器
}
catch (ClassNotFoundException e)
{
throw new JspException(noSuchHandler);
}
catch (InstantiationException e1)
{
throw new JspException(InstantiationErr);
}
catch (IllegalAccessException e2)
{
throw new JspException(accessErr);
}
return (EVAL_BODY_BUFFERED);
}
//我们规定页面的其他界面元素均在PageTag标签的范围内
//定义,则当PageTag.doEndTag调用时,也意味着页面
//执行完毕
final public int doEndTag() throws JspException
{
//为PageHandler增加一个可重写的函数onUnload
//子类可以重写此函数完成页面执行完之后的工作
handler.onUnload();
return SKIP_PAGE;
}
}
我们还需要定义一个FormTag标签以取代html的Form,在她的实现中,要求能输出javaScript函数on_event的类似功能实现,自动将form的action指向本页面,此外,还要自动输出两个隐藏字段,分别用来持有页面发生的事件名称和需要向服务器传递的参数。
这样,我们具体的页面定义看起来象下面的样子:
<%@ page contentType="text/html; charset=GB2312" %>
<%@ taglib uri="/WEB-INF/myjsp-html.tld" prefix="myjsp" %>
<!--pageTag标签, 属性pageHandler指定了本页面的处理器-->
<myjsp:page pageHandler="myjsp.test.JspTest">
<html>
<head><title>测试页面处理器</title></head>
<body bgcolor="#ffffff">
<h1>测试页面处理器</h1>
<myjsp:form method="post" id="myForm"><!--FormTag标签,它帮我们自动生成许多代码-->
请输入数字:<input type="text" name="t_value" value="<%=handler.t_value%>">
<br><br>
<font color="red"><%=handler.result%></font>
<br><br>
<input type="button" name="b1" value="2倍" onclick="on_event('onTwo','')">
<input type="button" name="b2" value="10倍" onclick="on_event('onTen','')">
</myjsp:form>
</body>
</html>
</myjsp:page>
至此,我们简化了 PageHandler使用。这样的模式使得页面具有了明显的生命周期(体现在PageHandler中):
onLoad():PageHandler的子类可在此方法中进行页面的初始化,获取页面参数并作初步处理。
服务器端事件:如JspTest中定义的onTwo和onTen,如果在客户端触发了相应的事件的话,它们将在onLoad之后执行。
onUnload():页面执行完毕后被调用,可以作页面持有资源的清理工作。
在客户端的每一次请求中(包括回调),页面处理器都会按照onLoad、服务器端事件、onUnload的流程顺序运作。把握了这一点,就能很好地组织自己的代码,写出功能复杂的页面来。
这类似WebForm的运作方式,同样,要很好运用asp.net,也必须注意到WebForm的运作流程。很多习惯于开发桌面应用的人在开始用WebForm时,会随意地说:太简单了,和我以前开发 CS没什么分别,孰不知当用户点击了WebForm中的一个按钮而引起回送时,不但事件函数会执行,onLoad和onUnload也将一前一后地又被调用(为什么说又,因为在回调之前页面初次被请求时,也肯定已经调 用了onLoad和onUnload)。而桌面应用中的窗体,在其显现时调用一次onLoad,关闭时调用onUnload,中间用户触发了事件只会调用对应的事件处理函数。道理虽简单,不知道的话却要吃大亏。
如果在这样的模型基础上,定义一套自定义标签集合,则能在Jsp中实现类似WebForm的功能。当然,我们的前提是不能改变现有jsp,servlet容器的运行方式。
(待续...)
相关文章: