在Jsp中模拟WebForm(二)

类别:Java 点击:0 评论:0 推荐:

                                                     在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','')">
        &nbsp;&nbsp;&nbsp;
        <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(五)

本文地址:http://com.8s8s.com/it/it16657.htm