一个大家不常用到,却很有用的页面间传值方法(Context.Handler) (转)
目前在ASP.NET中页面传值共有这么几种方式:
1、表单提交,
<form action= "target.aspx" method = "post" name = "form1">
<input name = "param1" value = "1111"/>
<input name = "param2" value = "2222"/>
</form>
....
form1.submit();
....
此种方在ASP。NET中无效,因为ASP。NET的表单总是提交到自身页面,如果要提交到别一页面,需要特殊处理。
2、<A href="target.aspx?param1=1111¶m2=2222">链接地址传送</A>
接收页面: string str = Request["param1"]
3、Session共享
发送页面:Session("param1") = "1111";
按收页面 string str = Session("param1").ToString();
4、Application共享
发送页面: Application("param1") = "1111";
按收页面: string str = Application("param1").ToString();
此种方法不常使用,因为Application在一个应用程序域范围共享,所有用户可以改变及设置其值,故只应用计数器等需要全局变量的地方。
5、Cookie
6、Response.Redirect()方式
Response.Redirect("target.aspx?param1=1111¶m2=2222")
接收页面: string str = Request["param1"]
7、Server.Transfer()方式。
Server.Transfer("target.aspx?param1=1111¶m2=2222")
接收页面: string str = Request["param1"]
二、如果在两个页面间需要大量的参数要传传递,如数据查询等页面时,用1 - 6的方法传值及其不便,而第 7 种方法确有一独特的优势!但使用该方法时需要一定的设置,现简单介绍一下该方法的使用方式:
以查询数据页面为例:
在查询页面中设置如下公有属性(QueryPage.aspx):
public class QueryPage : System.Web.UI.Page
...{
protected System.Web.UI.WebControls.TextBox txtStaDate;
protected System.Web.UI.WebControls.TextBox txtEndDate;
...
/**//// <summary>
/// 开始时间
/// </summary>
public string StaDate
...{
get...{ return this.txtStaDate.Text;}
set...{this.txtStaDate.Text = value;}
}
/**//// <summary>
/// 结束时间
/// </summary>
public string EndDate
...{
get...{ return this.txtEndDate.Text;}
set...{this.txtEndDate.Text = value;}
}
....
private void btnEnter_Click(object sender, System.EventArgs e)
...{
Server.Transfer("ResultPage.aspx"); //注意:使用ResultPage.aspx来接收传递过来的参数
}
}
在显示查询结果页面(ResultPage.aspx):
public class ResultPage : System.Web.UI.Page
...{
private void Page_Load(object sender, System.EventArgs e)
...{
//转换一下即可获得前一页面中输入的数据
QueryPage queryPage = ( QueryPage )Context.Handler; //注意:引用页面句柄
Response.Write( "StaDate:" );
Response.Write( queryPage.StaDate );
Response.Write( "<br/>EndDate:" );
Response.Write( queryPage.EndDate );
}
}
三、如果有许多查询页面共用一个结果页面的设置方法:
在这种方式中关键在于“ QueryPage queryPage = ( QueryPage )Context.Handler; ”的转换,只有转换不依赖于特定的页面时即可实现。
如果让所有的查询页面都继承一个接口,在该接口中定义一个方法,该方法的唯一作用就是让结果页面获得构建结果时所需的参数,就可实现多页面共享一个结果页面操作!
1、先定义一个类,用该类放置所有查询参数:(*.cs)
/**//// <summary>
/// 结果页面中要用到的值
/// </summary>
public class QueryParams
...{
private string staDate;
private string endDate;
/**//// <summary>
/// 开始时间
/// </summary>
public string StaDate
...{
get...{ return this.staDate;}
set...{this.staDate = value;}
}
/**//// <summary>
/// 结束时间
/// </summary>
public string EndDate
...{
get...{ return this.endDate;}
set...{this.endDate = value;}
}
}
2、接口定义:
/**//// <summary>
/// 定义查询接口。
/// </summary>
public interface IQueryParams
...{
/**//// <summary>
/// 参数
/// </summary>
QueryParams Parameters...{get;}
}
3、查询页面继承IQueryParams接口(QueryPage.aspx):
/**//// <summary>
///查询页面,继承接口
/// </summary>
public class QueryPage : System.Web.UI.Page, IQueryParams
...{
protected System.Web.UI.WebControls.TextBox txtStaDate;
protected System.Web.UI.WebControls.TextBox txtEndDate;
private QueryParams queryParams;
...
/**//// <summary>
/// 结果页面用到的参数
/// </summary>
public QueryParams Parameters
...{
get
...{
return queryParams;
}
}
....
private void btnEnter_Click(object sender, System.EventArgs e)
...{
//赋值
queryParams = new QueryParams();
queryParams.StaDate = this.txtStaDate.Text;
queryParams.EndDate = this.txtEndDate.Text
Server.Transfer("ResultPage.aspx");
}
}
4、别外的页面也如此设置
5、接收页面(ResultPage.aspx):
public class ResultPage : System.Web.UI.Page
...{
private void Page_Load(object sender, System.EventArgs e)
...{
QueryParams queryParams = new QueryParams();
IQueryParams queryInterface;
//实现该接口的页面
if( Context.Handler is IQueryParams)
...{
queryInterface = ( IQueryParams )Context.Handler;
queryParams = queryInterface.Parameters;
}
Response.Write( "StaDate:" );
Response.Write( queryParams.StaDate );
Response.Write( "<br/>EndDate:" );
Response.Write( queryParams.EndDate );
}
}
三、本文起因:
因在工作中要作一个数据查询,参数烦多,原先是用Session传递,用完该Session传来的参数后,还需清理Session,在用Session之前还得判断该Session是否存在,极其烦琐,我想应该还有更简便的方法来实现页面间的参数传递,故上网查找,终于找到这样一种方式来实现页面间的参数传递。
有不到之处,请大家指正!
==================================================================================
首先来看HttpContext类:
System.Web.HttpContext类继承自System.Object,按类名来理解,即是Http上下文类.
此类封装了有关单个HTTP 请求的所有HTTP 特定的信息。此类为继承 IHttpModule 和 IHttpHandler 接口的类提供了对当前 HTTP 请求的 HttpContext 对象的引用。该对象提供对请求的内部 Request、Response 和 Server 对象的访问。
HttpContext类的常用公共属性有:
Application,为当前 HTTP 请求获取HttpApplicationState 对象。
Current, 为当前 HTTP 请求获取 HttpContext 对象。
Handler, 为当前 HTTP 请求获取或设置 IHttpHandler 对象。
Items,获取可用于在 HTTP 请求过程中在 IHttpModule 和 IHttpHandler 之间组织和共享数据的键值集合。
Request,为当前 HTTP 请求获取 HttpRequest 对象。
Response,为当前 HTTP 响应获取 HttpResponse 对象。
Server, 获取提供用于处理 Web 请求的方法的 HttpServerUtility 对象。
Session, 为当前 HTTP 请求获取 HttpSessionState 实例。
通过Page类的Context属性可以获得当前的System.Web.HttpContext对象
接着来看Server.Transer()方法:
通过Page类的Server属性类可以Transfer到另一个页面,如Server.Transfer("NewPage.aspx"),可以跳转到新页面中,
使用Server.Transfer()跳转页面,客户端的URL并不会改变,只是在服务器端执行新页并输出,因此可以在新页面中通过获取来获得请求页面传递的对象和表单数据及查询字符串.
假定当前页面为FormerPage.aspx(类名为FormerPage), 跳转到的新页面为NewPage.aspx
从FormerPage.aspx跳转的代码如下:
private void btnToNewPage_Click(object sender, System.EventArgs e)
{
ArrayList list = new ArrayList(3);
list.Add("This list ");
list.Add("is for ");
list.Add("FormerPage to see.");
Context.Items["FormerPageList1"] = list;
Server.Transfer("NewPage.aspx");
}
在新页面(NewPage.aspx)Page_Load()事件中通过如下代码获得传递的数据:
if(!IsPostBack)
{
try
{
FormerPage former = (FormerPage)Context.Handler;
txtFromParentPage.Text = former.ClassName; //获取FormerPage中定义的ClassName公共属性
//获取在FormerPage中的上下文字典中添加的ArrayList: Context.Items["FormerPageList1"]
//获取Contex字典项并强制转换类型:
ArrayList list = Context.Items["FormerPageList1"] as ArrayList;
DataSet ds = former.GetDataSet(); //调用FormerPage中定义的GetDataSet()公共方法
DataGrid1.DataSource = ds;
DataGrid1.DataBind();
}
catch
{
Response.Write("Error get data from parent transfer page!");
}
}
注意上面通过Context.Hander属性来获取当前Http请求的IHttpHander对象,并强制转换成FormerPage对象:
FormerPage former = (FormerPage)Context.Handler;
后面可以直接调用这个类的公共属性,方法.同时可以调用在FormerPage中添加的Context字典项(Dictionary Item).
值得注意的是,使用Server.Transer传递页面数据,并使用Context.Handler来接收数据,只有在页面首次加载时,才可以正确获取上一页面的实例,而在postback时,获取的就会是当前页面的实例.
比如在NewPage.aspx中,首次加载时可以获取FomerPage对象,而在回发时尝试获取FormerPage会抛出异常,因为回发时,请求页面已经发生改变,不再是FormerPage发出的请求,而是自身NewPage发出的请求.我们可以在Page_Load()中加入如下代码判断Http 请求是由哪个页面发出的:
string path = Context.Request.Path;
Response.Write("<script>alert('Request from:" + path + "');</script>");
另外,Server.Transer()有一个重载的方法Server.Trasfer(string newpage,bool preserveForm), 第二个参数用来指定是否保留HttpRequest.Form和HttpRequest.QueryString集合,若为true, 则原页面的Form和QueryString在新的页面中依然有效,可以被调用. 如:
string str = "Value of Textbox:"+Request.Form["TextBox1"] +"<br>";
1、表单提交,
<form action= "target.aspx" method = "post" name = "form1">
<input name = "param1" value = "1111"/>
<input name = "param2" value = "2222"/>
</form>
....
form1.submit();
....
此种方在ASP。NET中无效,因为ASP。NET的表单总是提交到自身页面,如果要提交到别一页面,需要特殊处理。
2、<A href="target.aspx?param1=1111¶m2=2222">链接地址传送</A>
接收页面: string str = Request["param1"]
3、Session共享
发送页面:Session("param1") = "1111";
按收页面 string str = Session("param1").ToString();
4、Application共享
发送页面: Application("param1") = "1111";
按收页面: string str = Application("param1").ToString();
此种方法不常使用,因为Application在一个应用程序域范围共享,所有用户可以改变及设置其值,故只应用计数器等需要全局变量的地方。
5、Cookie
6、Response.Redirect()方式
Response.Redirect("target.aspx?param1=1111¶m2=2222")
接收页面: string str = Request["param1"]
7、Server.Transfer()方式。
Server.Transfer("target.aspx?param1=1111¶m2=2222")
接收页面: string str = Request["param1"]
二、如果在两个页面间需要大量的参数要传传递,如数据查询等页面时,用1 - 6的方法传值及其不便,而第 7 种方法确有一独特的优势!但使用该方法时需要一定的设置,现简单介绍一下该方法的使用方式:
以查询数据页面为例:
在查询页面中设置如下公有属性(QueryPage.aspx):
public class QueryPage : System.Web.UI.Page
...{
protected System.Web.UI.WebControls.TextBox txtStaDate;
protected System.Web.UI.WebControls.TextBox txtEndDate;
...
/**//// <summary>
/// 开始时间
/// </summary>
public string StaDate
...{
get...{ return this.txtStaDate.Text;}
set...{this.txtStaDate.Text = value;}
}
/**//// <summary>
/// 结束时间
/// </summary>
public string EndDate
...{
get...{ return this.txtEndDate.Text;}
set...{this.txtEndDate.Text = value;}
}
....
private void btnEnter_Click(object sender, System.EventArgs e)
...{
Server.Transfer("ResultPage.aspx"); //注意:使用ResultPage.aspx来接收传递过来的参数
}
}
在显示查询结果页面(ResultPage.aspx):
public class ResultPage : System.Web.UI.Page
...{
private void Page_Load(object sender, System.EventArgs e)
...{
//转换一下即可获得前一页面中输入的数据
QueryPage queryPage = ( QueryPage )Context.Handler; //注意:引用页面句柄
Response.Write( "StaDate:" );
Response.Write( queryPage.StaDate );
Response.Write( "<br/>EndDate:" );
Response.Write( queryPage.EndDate );
}
}
三、如果有许多查询页面共用一个结果页面的设置方法:
在这种方式中关键在于“ QueryPage queryPage = ( QueryPage )Context.Handler; ”的转换,只有转换不依赖于特定的页面时即可实现。
如果让所有的查询页面都继承一个接口,在该接口中定义一个方法,该方法的唯一作用就是让结果页面获得构建结果时所需的参数,就可实现多页面共享一个结果页面操作!
1、先定义一个类,用该类放置所有查询参数:(*.cs)
/**//// <summary>
/// 结果页面中要用到的值
/// </summary>
public class QueryParams
...{
private string staDate;
private string endDate;
/**//// <summary>
/// 开始时间
/// </summary>
public string StaDate
...{
get...{ return this.staDate;}
set...{this.staDate = value;}
}
/**//// <summary>
/// 结束时间
/// </summary>
public string EndDate
...{
get...{ return this.endDate;}
set...{this.endDate = value;}
}
}
2、接口定义:
/**//// <summary>
/// 定义查询接口。
/// </summary>
public interface IQueryParams
...{
/**//// <summary>
/// 参数
/// </summary>
QueryParams Parameters...{get;}
}
3、查询页面继承IQueryParams接口(QueryPage.aspx):
/**//// <summary>
///查询页面,继承接口
/// </summary>
public class QueryPage : System.Web.UI.Page, IQueryParams
...{
protected System.Web.UI.WebControls.TextBox txtStaDate;
protected System.Web.UI.WebControls.TextBox txtEndDate;
private QueryParams queryParams;
...
/**//// <summary>
/// 结果页面用到的参数
/// </summary>
public QueryParams Parameters
...{
get
...{
return queryParams;
}
}
....
private void btnEnter_Click(object sender, System.EventArgs e)
...{
//赋值
queryParams = new QueryParams();
queryParams.StaDate = this.txtStaDate.Text;
queryParams.EndDate = this.txtEndDate.Text
Server.Transfer("ResultPage.aspx");
}
}
4、别外的页面也如此设置
5、接收页面(ResultPage.aspx):
public class ResultPage : System.Web.UI.Page
...{
private void Page_Load(object sender, System.EventArgs e)
...{
QueryParams queryParams = new QueryParams();
IQueryParams queryInterface;
//实现该接口的页面
if( Context.Handler is IQueryParams)
...{
queryInterface = ( IQueryParams )Context.Handler;
queryParams = queryInterface.Parameters;
}
Response.Write( "StaDate:" );
Response.Write( queryParams.StaDate );
Response.Write( "<br/>EndDate:" );
Response.Write( queryParams.EndDate );
}
}
三、本文起因:
因在工作中要作一个数据查询,参数烦多,原先是用Session传递,用完该Session传来的参数后,还需清理Session,在用Session之前还得判断该Session是否存在,极其烦琐,我想应该还有更简便的方法来实现页面间的参数传递,故上网查找,终于找到这样一种方式来实现页面间的参数传递。
有不到之处,请大家指正!
==================================================================================
首先来看HttpContext类:
System.Web.HttpContext类继承自System.Object,按类名来理解,即是Http上下文类.
此类封装了有关单个HTTP 请求的所有HTTP 特定的信息。此类为继承 IHttpModule 和 IHttpHandler 接口的类提供了对当前 HTTP 请求的 HttpContext 对象的引用。该对象提供对请求的内部 Request、Response 和 Server 对象的访问。
HttpContext类的常用公共属性有:
Application,为当前 HTTP 请求获取HttpApplicationState 对象。
Current, 为当前 HTTP 请求获取 HttpContext 对象。
Handler, 为当前 HTTP 请求获取或设置 IHttpHandler 对象。
Items,获取可用于在 HTTP 请求过程中在 IHttpModule 和 IHttpHandler 之间组织和共享数据的键值集合。
Request,为当前 HTTP 请求获取 HttpRequest 对象。
Response,为当前 HTTP 响应获取 HttpResponse 对象。
Server, 获取提供用于处理 Web 请求的方法的 HttpServerUtility 对象。
Session, 为当前 HTTP 请求获取 HttpSessionState 实例。
通过Page类的Context属性可以获得当前的System.Web.HttpContext对象
接着来看Server.Transer()方法:
通过Page类的Server属性类可以Transfer到另一个页面,如Server.Transfer("NewPage.aspx"),可以跳转到新页面中,
使用Server.Transfer()跳转页面,客户端的URL并不会改变,只是在服务器端执行新页并输出,因此可以在新页面中通过获取来获得请求页面传递的对象和表单数据及查询字符串.
假定当前页面为FormerPage.aspx(类名为FormerPage), 跳转到的新页面为NewPage.aspx
从FormerPage.aspx跳转的代码如下:
private void btnToNewPage_Click(object sender, System.EventArgs e)
{
ArrayList list = new ArrayList(3);
list.Add("This list ");
list.Add("is for ");
list.Add("FormerPage to see.");
Context.Items["FormerPageList1"] = list;
Server.Transfer("NewPage.aspx");
}
在新页面(NewPage.aspx)Page_Load()事件中通过如下代码获得传递的数据:
if(!IsPostBack)
{
try
{
FormerPage former = (FormerPage)Context.Handler;
txtFromParentPage.Text = former.ClassName; //获取FormerPage中定义的ClassName公共属性
//获取在FormerPage中的上下文字典中添加的ArrayList: Context.Items["FormerPageList1"]
//获取Contex字典项并强制转换类型:
ArrayList list = Context.Items["FormerPageList1"] as ArrayList;
DataSet ds = former.GetDataSet(); //调用FormerPage中定义的GetDataSet()公共方法
DataGrid1.DataSource = ds;
DataGrid1.DataBind();
}
catch
{
Response.Write("Error get data from parent transfer page!");
}
}
注意上面通过Context.Hander属性来获取当前Http请求的IHttpHander对象,并强制转换成FormerPage对象:
FormerPage former = (FormerPage)Context.Handler;
后面可以直接调用这个类的公共属性,方法.同时可以调用在FormerPage中添加的Context字典项(Dictionary Item).
值得注意的是,使用Server.Transer传递页面数据,并使用Context.Handler来接收数据,只有在页面首次加载时,才可以正确获取上一页面的实例,而在postback时,获取的就会是当前页面的实例.
比如在NewPage.aspx中,首次加载时可以获取FomerPage对象,而在回发时尝试获取FormerPage会抛出异常,因为回发时,请求页面已经发生改变,不再是FormerPage发出的请求,而是自身NewPage发出的请求.我们可以在Page_Load()中加入如下代码判断Http 请求是由哪个页面发出的:
string path = Context.Request.Path;
Response.Write("<script>alert('Request from:" + path + "');</script>");
另外,Server.Transer()有一个重载的方法Server.Trasfer(string newpage,bool preserveForm), 第二个参数用来指定是否保留HttpRequest.Form和HttpRequest.QueryString集合,若为true, 则原页面的Form和QueryString在新的页面中依然有效,可以被调用. 如:
string str = "Value of Textbox:"+Request.Form["TextBox1"] +"<br>";