在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容器的运行方式。       (待续...)
  相关文章: 在Jsp中模拟WebForm(一) 在Jsp中模拟WebForm(三) 在Jsp中模拟WebForm(四) 在Jsp中模拟WebForm(五)
  
 
  |