分享
 
 
 

在JSP开发中模拟.NET WebForm

王朝java/jsp·作者佚名  2006-11-24
窄屏简体版  字體: |||超大  

WebForm是事件驱动的,控件状态可以在http请求之间自动保持,并且使用后置代码很好地实现了页面外观与页面逻辑控制的分离,一改以往html,服务器段代码、javaScript混杂在一起的web开发方式。stucts提供了大量的定制标签,由tag、form、bean、action及配置文件构建了一个优秀的MVC模式的web开发方式。但相比较其WebForm来,窃以为stucts更为复杂,需要协同工作的元素较多,解决问题的效果不如WebForm显著(仅是个人看法)。

在现实开发中,常常需要在某个页面中处理很多Form控件,且要处理这个页面可能引发的多个事件,在事件触发后,又请求同一个页面,又需要在请求之间保持状态,在页面中处理所有这些,真实不胜其烦。受到WebForm启发,我在用JSP进行开发时,借鉴了了其一些思想。本质上我们就是想让页面显示代码与页面控制代码分离,要作到这一点并不困难,有很多办法。

可以为页面定义一个“页面处理器(PageHandler)”,它类似WebForm的后置代码,它的接口基本是下面这个样子:

public class PageHandler

{

protected HttpServletRequest request;

protected HttpServletResponse response;

protected JspWriter out;

protected PageContext pageContext;

protected HttpSession session = null;

protected ServletContext application = null;

protected ServletConfig config = null;

protected String event_action = null; //页面事件

protected String event_params = null; //页面参数

//取得操作页面的基本组件

public PageHandler(PageContext page)

{

this.pageContext = page;

this.request = (HttpServletRequest) pageContext.getRequest();

this.response = (HttpServletResponse) pageContext.getResponse();

this.pageContext = page;

out = pageContext.getOut();

application = pageContext.getServletContext();

config = pageContext.getServletConfig();

session = pageContext.getSession();

try{

request.setCharacterEncoding("gb2312");//设定页面编码

}

catch(Exception e)

{

e.printStackTrace();

}

}

//初始化页面的参数,具体的页面处理器类可以重写这

//个方法进行页面初始化

protected void onLoad() throws Exception

{

}

//根据页面指定的事件进行处理

private final void eventBind() throws Exception

{

//event_action从从页面的名为event_action的hidden字段取得,它意为事件的称,

//当此事件触发时,他会寻找在"页面处理器类中"与event_action同名的方法加

// 以调用。

if (event_action != null && !event_action.equals(Format.Empty))

{

event_params = request.getParameter("parameters"); //事件参数参数,从页面

//的名为parameters的hidden字段取得

if (paramTypes[0] == null)

{

paramTypes[0] = Class.forName("java.lang.String");

}

Object paramValues[] = new Object[1];

paramValues[0] = event_params;

Method method = null;

try

{

method = this.getClass().getDeclaredMethod(event_action, paramTypes);

method.setAccessible(true);

}

catch (Exception e)

{

throw new UserException("系统缺少对您的请求的处理机制: + event_action);

}

if (method != null)

{

method.invoke(this, paramValues); //调用web时间

}

}

}

//处理页面

public void process() throws Exception

{

try

{

event_action = request.getParameter("action"); //得页面事件

onLoad();//页面加载时的初始化

eventBind();//处理事件

}

catch (Exception e)

{

e.printStackTrace(); ///////////////

Format.alert(out, "发生了未知错误:" + Format.getString(e.getMessage()));

}

}

}

当然,实用的 PageHandler应提供更为复杂的功能。

具体的页面处理器类从此类继承下来,现在,我们用一个简单的例子说明页面处理器的用法:假设有这样一个页面,有一个文本框要求用户输入一个数字,有两个按钮,点击一个要求计算出用户输入数字的2倍,点击另外一个按钮要求计算出用户输入数字的10倍。再假设此页面的页面处理器类为JspTest.

//test.jsp

<%@ page contentType="text/html; charset=GB2312" %>

<%@ page import="youpackage.JspTest" %>

<%

JspTest handler=new JspTest(pageContext);

handler.process();//调用页面处理器

String formAction=request.getRequestURI()+"?"+request.getQueryString();

%>

<html>

<head>

<title>测试页面处理器</title>

<script language="javascript">

function on_event(action,params)

{

window.form1.action.value=action;

window.form1.parameters.value=params;

window.form1.submit();

}

</script>

</head>

<body bgcolor="#ffffff">

<form name="form1" method="post" action="<%=formAction%>">

请输入数字:<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','')">

<input type="hidden" name="action" value=""/>

<input type="hidden" name="parameters" value=""/>

</form>

</body>

</html>

则,我们为以上页面定义其页面处理器:JspTest

//JspTest.java

public class JspTest extends PageHandler

{

//定义页面变量

public int t_value;//用户输入的整数

public String result;//存储计算结果

public JspTest(PageContext page)

{

super(page);

}

protected void onLoad() throws Exception

{

t_value=0;

result="";

//在实际应用中,这里应作许多的初始化工作(如,得到页面参数)

}

//双倍

private void onTwo(String str_params) throws Exception

{

try

{

t_value=Integer.parseInt(request.getParameter("t_value"));

}

catch(Exception e)

{

out.println("<script language='javaScript'>alert('您输入的不是有效的整数.');</script>");

}

int i=2*t_value;

result="计算结果为:"+i;

}

//10倍

private void onTen(String str_params) throws Exception

{

try

{

t_value=Integer.parseInt(request.getParameter("t_value"));

}

catch(Exception e)

{

out.println("<script language='javaScript'>alert('您输入的不是有效的整数.');</script>");

}

int i=10*t_value;

result="计算结果为:"+i;

}

}

WebForm的基本思想也就在于此,当然,WebForm中的服务器端控件的状态可以自动保持(而我们的实现为保持状态还需作一些工作),WebForm的控件属性可以在后置代码中进行操作,服务器端事件可以在后置代码中进行邦定,服务器端控件支持数据邦定等等,我们的实现还无法做到。

如果能在Jsp中定义类似服务器端控件的东东,以上的功能在Jsp中可以得以实现。

用“页面处理器”的方式组织页面代码,起到了将页面显示元素与服务器端控制代码分离的目的,使得我们的代码更为清晰。在页面上,需要例行公事地调用页面相应的处理器(多个页面可以具有相同的处理器),声明一个类似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"><%=hand

[1] [2] 下一页

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
推荐阅读
 
 
 
>>返回首頁<<
 
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有