解决刷新页面造成的数据重复提交问题

执行过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。 SubmitOncePage SubmitOncePage是针对上述分析写的一个继承自System.Web.UI.Page的基类,需要防止刷新重复提交数据的页面从该基类继承,源码如下: namespace myControl { /// /// 名称:SubmitOncePage /// 父类:System.Web.UI.Page /// 描述:解决浏览器刷新造成的数据重复提交问题的page扩展类。 /// 示例: if (!this.IsRefreshed) /// { /// //具体代码 /// } /// 原创:丛兴滋(cncxz) E-mail:cncxz@126.com /// public class SubmitOncePage:System.Web.UI.Page { private string _strSessionKey; private string _hiddenfieldName; private string _strLastViewstate; public SubmitOncePage() { _hiddenfieldName = "__LastVIEWSTATE_SessionKey"; _strSessionKey = System.Guid.NewGuid().ToString(); _strLastViewstate = string.Empty; } public bool IsRefreshed { get { string str1 = GetSessinContent(); _strLastViewstate = str1; string str2 = this.Session[GetSessinKey()] as string; bool flag1 = (str1 != null) && (str2 != null) && (str1 == str2); return flag1; } } protected override void Render(System.Web.UI.HtmlTextWriter writer) { string str = GetSessinKey(); this.Session[str] = _strLastViewstate; this.RegisterHiddenField(_hiddenfieldName, str); base.Render(writer); } private string GetSessinKey() { string str = this.Request.Form[_hiddenfieldName]; return (str == null) ? _strSessionKey : str; } private string GetSessinContent() { string str = this.Request.Form["__VIEWSTATE"]; if (str == null) { return null; } return System.Web.Security.FormsAuthentication.HashPasswordForStoringInConfigFile(str, "MD5"); } } }

转载于:https://www.cnblogs.com/YingLoveHaiTao/archive/2011/08/22/2148920.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值