SubmitOncePage:解决刷新页面造成的数据重复提交问题
2008-09-18 11:01
896 查看
执行过postback操作的web页面在刷新的时候,浏览器会有“不重新发送信息,则无法刷新网页”的提示,若刚刚执行的恰好是往数据库插入一条新记录的操作,点[重试]的结果是插入了两条重复的记录,以前一直是用保存数据后重新转向当前页面的方法解决,最近又找到了一个新的方法。
问题分析
在System.Web.UI.Page类中,有一个名为ViewState属性用以保存页面的当前视图状态,观察每个aspx页面最终生成的html代码可以发现,其实就是向页面添加了一个名为__VIEWSTATE的隐藏域,其value值就是页面的当前状态,每次执行postback过后,该value值都会发生变化,而刷新页面则不会改变。
针对这种情况,我们可以在页面代码执行的末尾将当前的ViewState写到一个Session中,而在页面加载时则判断该Session值是否与当前ViewState相等(其实Session值恰好是ViewState的前一状态),若不等,则是正常的postback,若是相等则是浏览器刷新,这样一来,只要在我们的数据插入代码外嵌套一个if判断就可以达到防止数据重复提交的目的了。
其实到这里问题还没有完全解决,具体说来就是Session的键值问题。假设我们将ViewState保存为this.Session["myViewState"],如果一个用户同时打开两个防刷新提交的页面就乱套了,那针对页面的url设置Session的键值呢?还是不行,因为用户有可能在两个窗口中打开同一页面,所以必须为每次打开的页面定义唯一的Session键值,并且该键值可以随当前页面实例一起保存,参考ViewState的保存方式,我们直接向页面添加一个隐藏域专门存放Session键值就可以了。
经oop80和Edward.Net的提醒,为了尽可能地降低Session数据对服务器资源的占用量,现将上述方案略做调整,将ViewState利用md5加密后返回的32位字符串写入Session。
另外,由于本方法会生成额外的Session占用服务器资源,所以请在必须保留当前页面状态的情况下使用,若无需保留当前页面状态,则在完成数据提交后直接重定向到当前页面即可。
SubmitOncePage
SubmitOncePage是针对上述分析写的一个继承自System.Web.UI.Page的基类,需要防止刷新重复提交数据的页面从该基类继承,源码如下:
复制
保存
测试项目
首先将SubmitOncePage类的源码编译成一个单独的dll,然后进行测试,步骤如下:
1、新建一个asp.net web应用程序;
2、添加SubmitOncePage类对应的dll引用;
3、给webform1添加一个Label控件(Label1)和一个Button控件(Button1);
4、设置Label1的Text为0;
5、双击Button1转到codebehind视图;
6、修改类WebForm1的父类为SubmitOncePage并添加测试代码,结果如下:
复制
保存
7、按F5运行,在浏览器窗口中连续点击几次Button1,然后刷新几次页面,再点击几次Button1;
8、转到测试项目对应目录下,打开a.txt和b.txt文件,可看到if (!this.IsRefreshed) 的具体效果。
相关文件下载
http://www.cnblogs.com/Files/cncxz/SubmitOncePage.rar
问题分析
在System.Web.UI.Page类中,有一个名为ViewState属性用以保存页面的当前视图状态,观察每个aspx页面最终生成的html代码可以发现,其实就是向页面添加了一个名为__VIEWSTATE的隐藏域,其value值就是页面的当前状态,每次执行postback过后,该value值都会发生变化,而刷新页面则不会改变。
针对这种情况,我们可以在页面代码执行的末尾将当前的ViewState写到一个Session中,而在页面加载时则判断该Session值是否与当前ViewState相等(其实Session值恰好是ViewState的前一状态),若不等,则是正常的postback,若是相等则是浏览器刷新,这样一来,只要在我们的数据插入代码外嵌套一个if判断就可以达到防止数据重复提交的目的了。
其实到这里问题还没有完全解决,具体说来就是Session的键值问题。假设我们将ViewState保存为this.Session["myViewState"],如果一个用户同时打开两个防刷新提交的页面就乱套了,那针对页面的url设置Session的键值呢?还是不行,因为用户有可能在两个窗口中打开同一页面,所以必须为每次打开的页面定义唯一的Session键值,并且该键值可以随当前页面实例一起保存,参考ViewState的保存方式,我们直接向页面添加一个隐藏域专门存放Session键值就可以了。
经oop80和Edward.Net的提醒,为了尽可能地降低Session数据对服务器资源的占用量,现将上述方案略做调整,将ViewState利用md5加密后返回的32位字符串写入Session。
另外,由于本方法会生成额外的Session占用服务器资源,所以请在必须保留当前页面状态的情况下使用,若无需保留当前页面状态,则在完成数据提交后直接重定向到当前页面即可。
SubmitOncePage
SubmitOncePage是针对上述分析写的一个继承自System.Web.UI.Page的基类,需要防止刷新重复提交数据的页面从该基类继承,源码如下:
复制
保存
namespace[/b] myControl { /// <summary> [/b] /// 名称:SubmitOncePage [/b] /// 父类:System.Web.UI.Page [/b] /// 描述:解决浏览器刷新造成的数据重复提交问题的page扩展类。 [/b] /// 示例: if (!this.IsRefreshed) [/b] /// { [/b] /// //具体代码 [/b] /// } [/b] /// 原创:丛兴滋(cncxz) E-mail:cncxz@126.com [/b] /// </summary> [/b] public[/b] class[/b] SubmitOncePage : System.Web.UI.Page { private[/b] string[/b] _strSessionKey; private[/b] string[/b] _hiddenfieldName; private[/b] string[/b] _strLastViewstate; public[/b] SubmitOncePage() { _hiddenfieldName = "__LastVIEWSTATE_SessionKey"[/b]; _strSessionKey = System.Guid.NewGuid().ToString(); _strLastViewstate = string[/b].Empty; } public[/b] bool[/b] IsRefreshed { get[/b] { string[/b] str1 = GetSessinContent(); _strLastViewstate = str1; string[/b] str2 = this[/b].Session[GetSessinKey()] as[/b] string[/b]; bool[/b] flag1 = (str1 != null[/b]) && (str2 != null[/b]) && (str1 == str2); return[/b] flag1; } } protected[/b] override[/b] void[/b] Render(System.Web.UI.HtmlTextWriter writer) { string[/b] str = GetSessinKey(); this[/b].Session[str] = _strLastViewstate; this[/b].RegisterHiddenField(_hiddenfieldName, str); base[/b].Render(writer); } private[/b] string[/b] GetSessinKey() { string[/b] str = this[/b].Request.Form[_hiddenfieldName]; return[/b] (str == null[/b]) ? _strSessionKey : str; } private[/b] string[/b] GetSessinContent() { string[/b] str = this[/b].Request.Form["__VIEWSTATE"[/b]]; if[/b] (str == null[/b]) { return[/b] null[/b]; } return[/b] System.Web.Security.FormsAuthentication.HashPasswordForStoringInConfigFile(str, "MD5"[/b]); } } }
测试项目
首先将SubmitOncePage类的源码编译成一个单独的dll,然后进行测试,步骤如下:
1、新建一个asp.net web应用程序;
2、添加SubmitOncePage类对应的dll引用;
3、给webform1添加一个Label控件(Label1)和一个Button控件(Button1);
4、设置Label1的Text为0;
5、双击Button1转到codebehind视图;
6、修改类WebForm1的父类为SubmitOncePage并添加测试代码,结果如下:
复制
保存
public[/b] class[/b] WebForm1 : myControl.SubmitOncePage { protected[/b] System.Web.UI.WebControls.Label Label1; protected[/b] System.Web.UI.WebControls.Button Button1; Web 窗体设计器生成的代码#region Web 窗体设计器生成的代码 override[/b] protected[/b] void[/b] OnInit(EventArgs e) { // [/b] // CODEGEN: 该调用是 ASP.NET Web 窗体设计器所必需的。 [/b] // [/b] InitializeComponent(); base[/b].OnInit(e); } /**/[/b]/// <summary> /// 设计器支持所需的方法 - 不要使用代码编辑器修改 [/b] /// 此方法的内容。 [/b] /// </summary> [/b] private[/b] void[/b] InitializeComponent() { this[/b].Button1.Click += new[/b] System.EventHandler(this[/b].Button1_Click); } #endregion [/b] private[/b] void[/b] Button1_Click(object[/b] sender, System.EventArgs e) { int[/b] i=int[/b].Parse(Label1.Text)+1; Label1.Text = i.ToString(); if[/b] (!this[/b].IsRefreshed) { WriteFile("a.txt"[/b], i.ToString()); } WriteFile("b.txt"[/b], i.ToString()); } private[/b] void[/b] WriteFile(string[/b] strFileName,string[/b] strContent) { string[/b] str = this[/b].Server.MapPath(strFileName); System.IO.StreamWriter sw = System.IO.File.AppendText(str); sw.WriteLine(strContent); sw.Flush(); sw.Close(); } }
7、按F5运行,在浏览器窗口中连续点击几次Button1,然后刷新几次页面,再点击几次Button1;
8、转到测试项目对应目录下,打开a.txt和b.txt文件,可看到if (!this.IsRefreshed) 的具体效果。
相关文件下载
http://www.cnblogs.com/Files/cncxz/SubmitOncePage.rar
相关文章推荐
- SubmitOncePage:解决刷新页面造成的数据重复提交问题
- SubmitOncePage:解决刷新页面造成的数据重复提交问题
- SubmitOncePage:解决刷新页面造成的数据重复提交问题
- SubmitOncePage:解决刷新页面造成的数据重复提交问题
- SubmitOncePage:解决刷新页面造成的数据重复提交问题
- SubmitOncePage:解决刷新页面造成的数据重复提交
- 解决刷新页面造成的数据重复提交问题
- 解决刷新页面造成的数据重复提交问题
- 解决浏览器刷新造成的数据重复提交问题的page扩展类。
- 解决刷新页面造成的数据重复提交问题
- 解决刷新页面造成的数据重复提交问题
- 解决刷新页面造成的数据重复提交问题
- 解决刷新页面造成的数据重复提交问题
- 解决后台提交表单数据时,重复F5刷新页面出现重复提交数据的问题。
- easyui关于解决tab添加的页面通过href加载数据内容的页面刷新后form表单重复提交的问题;
- 解决刷新页面造成数据重复提交的方法(转载)
- asp.net mvc利用Json验证数据和导向页面,解决重复提交问题
- 解决 php提交表单到当前页面,刷新会重复提交 的问题
- 用一个最简单方法解决asp.net页面刷新导致数据的重复提交
- JSP刷新页面表单重复提交问题解决办法分享