標準html表單(form元素)允許你向另外一個頁面或者應用程序傳遞和發送數據信息,方法是使用表單元素。在asp.net 1.x中,網頁則利用投遞機制,把頁面數據提交給該頁本身。對于asp.net 2.0,它的功能有所擴展,能夠允許跨頁提交。這周就讓我們來探討這個新特性。
傳統辦法
為了便于比較,我想花一分鐘來回顧網頁傳遞數據的老方法。html的表格元素有一個action(動作)屬性,用來指定服務器端哪項資源(所謂資源,是指一個網頁、一段腳本、程序等)來處理這些提交的數據。下面的代碼便是一個樣例。
<html>
<head><title>sample html form</title></head>
<body>
<form name="frmsample" method="post" action="target_url">
<input type="text" name="fullname" id="fullname" />
<input type="button" name="submit" value="submit" />
</form>
</body></html>
在文本域(名字是fullname)中輸入的值將被提交給表單元素的action屬性指定的頁面或者程序。對于asp.net開發者,即使曾經用過標準html表單,也是極不多見的。
asp.net開發者面對要從一個網頁向另一個網頁傳遞數據信息的任務時,方法選擇的余地是異常廣闊的。它們包括會話變量(session variables)、cookies、querystring 變量、caching(網頁緩存),甚至server.transfer方法,但是asp.net 2.0還提供了另外一種選擇。
asp.net 2.0提供的又一辦法
在設計asp.net 2.0的時候,微軟認識到了在網頁間交叉傳遞數據的需求。有了這個意識之后,就為asp.net的button(按鈕)控件增加了一項postbackurl屬性。它允許你在用戶提交的時候,指明這張表單和上面的數據送往何方(也就是由postbackurl屬性指定的url值確定)。一般來講,跨頁傳遞是客戶端在后臺使用javascript進行傳送的過程。
<%@ page language="vb" %>
<!doctype html public "-//w3c//dtd html 4.0 transitional//en" >
<html><head>
<title>cross postback example</title>
</head><body>
<form id="frmcrosspostback1" method="post" runat="server">
<asp:label id="lblname" runat="server" text="name:"></asp:label>
<asp:textbox id="txtname" runat="server"></asp:textbox><br />
<asp:label id="lble-mailaddress" runat="server" text="e-mail:"></asp:label>
<asp:textbox id="txte-mailaddress" runat="server"></asp:textbox><br />
<asp:button id="btnsubmit" runat="server" text="submit" postbackurl="crosspostback2.aspx" />
</form></body></html>
中的asp.net頁面擁有兩個文本域(分別表示name(名字)和e-mail(電子郵件)),以及一個用來提交數據的button(按鈕)。這個提交按鈕的postbackurl屬性被指定為另外一個網頁,這樣使得表單提交的時候,數據可以發送到那個頁面。注意:這個例子中,表單元素通過設置method(方法)屬性,讓表單提交時采用post[2]提交方式,但這不是必要的,因為所有cross postback(跨頁投遞)根據設計均使用post方法。
使用先前頁面
asp.net頁面經由跨頁投遞的調用而載入的時候,它上面的對象的ispostback屬性不會被觸發。不過,有一項叫做previouspage(前一頁)的屬性使你能夠訪問和使用那些應用跨頁投遞的頁面。
每當一個跨頁請求發生時,當前頁的previouspage屬性就把促發投遞的頁面引用保存下來。如果頁面的產生不是來自跨頁投遞的激發,或者說頁面處于不同的程序組,那么previouspage屬性將不會被初始化。
你可以通過檢查previouspage對象來確定頁面的載入是否為跨頁投遞的結果。值如果為null,則說明是普通的載入,而非null值則表明網頁來自跨頁投遞。此外,頁面類(page class)還包含了一個稱作iscrosspagepostback的方法(method),專門用來確定頁面是不是跨頁投遞的結果。
一旦確定發生了跨頁投遞,你就可以通過previouspage對象的findcontrol方法去訪問調用頁(calling page)上的控件。下面的代碼是我們例子中的第二頁;它由前面列出的頁面所調用。
<%@ page language="vb" %>
<!doctype html public "-//w3c//dtd html 4.0 transitional//en" >
<html><head>
<title>cross postback example 2</title>
</head><body>
<script language="vb" runat="server">
sub page_load()
if not (page.previouspage is nothing) then
if not (page.iscrosspagepostback) then
response.write("name:" + ctype(previouspage.findcontrol("txtname"), textbox).text + "<br>")
response.write("e-mail:" + ctype(previouspage.findcontrol("txte-mailaddress"), textbox).text + "<br>")
end if
end if
end sub
</script></body></html>
這個頁先判斷它是不是由跨頁投遞所調用。如果是,就通過findcontrol方法訪問來自調用頁的數值,并把用此方法得到的控件轉換為textbox控件,然后顯示它們的text(文本)屬性的內容。
你可以把整個previouspage對象轉換成觸發跨頁投遞的頁面類型。這個方法允許你訪問頁面的全局屬性(public properties)和方法。在我給出這項技術的實例之前,我有必要重寫第一個例子,包含進一些全局屬性。下面代碼是添加了兩個屬性的第一個清單,這兩個屬性用于訪問域值。
<%@ page language="vb" %>
<!doctype html public "-//w3c//dtd html 4.0 transitional//en" >
<html><head>
<title>cross postback example</title>
<script language="vb" runat="server">
public readonly property name
get
return me.txtname.text
end get
end property
public readonly property e-mailaddress
get
return me.txte-mailaddress.text
end get
end property
</script></head><body>
<form id="frmcrosspostback1" method="post" runat="server">
<asp:label id="lblname" runat="server" text="name:"></asp:label>
<asp:textbox id="txtname" runat="server"></asp:textbox><br />
<asp:label id="lble-mailaddress" runat="server" text="e-mail:"></asp:label>
<asp:textbox id="txte-mailaddress" runat="server"></asp:textbox><br />
<asp:button id="btnsubmit" runat="server" text="submit" postbackurl="crosspostback2.aspx" />
</form></body></html>
既然現在屬性已經建好,那你就能很容易訪問它們。要警惕的是,page類的previouspage對象必須轉換成正確的類型,這樣才能正確訪問它的屬性。這可以通過把它轉換成合適的page類別的對象加以實現。
<%@ page language="vb"%>
<%@ reference page="~/crosspostback1.aspx" %>
<!doctype html public "-//w3c//dtd html 4.0 transitional//en" >
<html><head>
<title>cross postback example 3</title>
</head><body>
<script language="vb" runat="server">
sub page_load()
dim cpppage as crosspostback1_aspx
if not (page.previouspage is nothing) then
if not (page.iscrosspagepostback) then
if (page.previouspage.isvalid) then
cpppage = ctype(previouspage, crosspostback1_aspx)
response.write("name:" + cpppage.name + "<br>")
response.write("e-mail:" + cpppage.e-mailaddress)
end if
end if
end if
end sub
</script></body></html>
說明了這一點,它在頁面頭部定義了調用頁的一項引用,那樣這個引用類型就能在代碼中使用。通過這項引用,實際的vb.net代碼使用ctype函數把previouspage對象轉換成了適當的類型。這之后,那些屬性就可以像代碼示范的那樣使用了。
關于上述清單中previouspage對象isvalid方法的使用在此提醒一下:前頁的isvalid屬性保證你對它操作之前,它已通過所有合法驗證測試。
總結
在網頁間傳遞數據參數有很多項應用,包括保持個人用戶信息。祖傳的網頁解決方案,像使用querystring和cookies,允許你很容易當提交發生時從一個頁面指向另一個頁面。
asp.net 1.1除了提供額外方法外,對這些方法也能很好地支持,可是,asp.net 2.0依靠跨頁投遞,使這方面又有了長足發展。它讓一個網頁處理來自另一網頁的數據變得簡單。在你開發你的下一個asp.net 2.0程序的時候,可要好好利用這個新概念的優勢啊。
新聞熱點
疑難解答