`
starbhhc
  • 浏览: 654302 次
  • 性别: Icon_minigender_2
  • 来自: 深圳
社区版块
存档分类
最新评论

彻底解决刷新重复提交问题

阅读更多
以前也研究过,始终没找到好的办法,看了微软Msdn上的解决方案,使用后发现存在较多漏洞,考虑的情况太少,如页面加载后没有提交,始终刷新,在同一浏览器打开多个各选项卡,每个选项卡打开同一页面或不同页面,以下是我的解决方案,
public class RefreshServe : System.Web.UI.Page
    {
        private static ILog log = LogManager.GetLogger(typeof(RefreshServe));

        private readonly string REFRESH_TICKET_NAME = "__RefreshTicketArray";
        private readonly string HIDDEN_FIELD_NAME = "__RefreshHiddenField";
        private readonly string HIDDEN_PAGE_GUID = "__RefreshPageGuid";

        /// <summary>
        /// 为True表示页面刷新,False为正常提交
        /// </summary>
        public bool IsPageRefreshed
        {
            get
            {
                if (IsPostBack && !CheckRefreshFlag())
                {
                    log.Debug("刷新了页面");
                    return true;
                }
                else
                {
                    log.Debug("正常提交");
                    return false;
                }
            }
        }

        /// <summary>
        /// 呈现前更新标识
        /// </summary>
        /// <param name="e"></param>
        protected override void OnPreRender(EventArgs e)
        {
            log.Debug("执行OnPreRender");
            base.OnPreRender(e);
            UpdateRefreshFlag();
        }


        /// <summary>
        /// 更新标识,正常提交都删除该次提交的时间,并生产当前新的时间
        /// </summary>
        private void UpdateRefreshFlag()
        {

            #region Session模式

            ////设置页面唯一标识
            //SetCurPageGuid();

            //DataTable RefreshSign = GetRefreshSession();
            //string pageGuid = GetCurPageGuid();
            //DataRow newRow;

            //if (RefreshSign.Rows.Count > 0)
            //{
            //    DataRow[] existRow = RefreshSign.Select("GUID='none'");
            //    if (existRow.Length > 0)
            //    {
            //        foreach (DataRow row in existRow)
            //            row.Delete();
            //        log.Debug("找到为none标识的行并删除");
            //    }

            //    existRow = RefreshSign.Select("GUID='" + pageGuid + "'");
            //    if (existRow.Length > 0)
            //    {
            //        foreach (DataRow row in existRow)
            //            row.Delete();
            //        log.Debug("找到为" + pageGuid + "标识的行并删除");
            //    }
            //}

            //string submitTime = DateTime.Now.ToString("hhmmss.fffff");
            ////当前提交时间保存到隐藏域
            //ClientScript.RegisterHiddenField(HIDDEN_FIELD_NAME, submitTime);


            ////同时添加到DataTable列表中
            //newRow = RefreshSign.NewRow();
            //newRow["submitTime"] = submitTime;
            //newRow["GUID"] = pageGuid;
            //log.Debug("即将要新增的票证:submitTime:" + submitTime + "  Guid:" + pageGuid.ToString());
            //RefreshSign.Rows.Add(newRow);

            //log.Debug("UpdateRefreshFlag中当前DataTable中存在的记录数为:" + RefreshSign.Rows.Count);
            //foreach (DataRow row in RefreshSign.Rows)
            //{
            //    log.Info("row['submitTime']:" + row["submitTime"] + "    row['GUID']:" + row["GUID"]);
            //}
            #endregion

            #region Cookie模式

            //注册页面唯一标识并返回
            string pageGuid = SetCurPageGuid();

            HttpCookie cookie = GetRefreshTicket();
          
            if (cookie.Values.Count > 0)
            {
                cookie.Values.Remove(pageGuid);
                log.Debug("当前清除的cookie变是:" + pageGuid);
            }

            string submitTime = DateTime.Now.ToString("hhmmss.fffff");
            //当前提交时间保存到隐藏域
            ClientScript.RegisterHiddenField(HIDDEN_FIELD_NAME, submitTime);


            log.Debug("即将要新增的时间:submitTime:" + submitTime + "  Guid:" + pageGuid.ToString());
            cookie.Values.Add(pageGuid, submitTime);

            log.Debug("UpdateRefreshFlag中当前Cookie中存在的记录数为:" + cookie.Values.Count);
            for (int i = 0; i < cookie.Values.Count; i++)
                log.Info("cookie[" + cookie.Values.GetKey(i) + "]:" + cookie.Values[i]);

            Response.AppendCookie(cookie);

            #endregion

        }


        /// <summary>
        /// 验证是否刷新
        /// </summary>
        /// <returns></returns>
        private bool CheckRefreshFlag()
        {
            #region Session模式
            //DataTable RefreshSign = GetRefreshSession();

            //if (RefreshSign.Rows.Count == 0)//第一次访问页面
            //{
            //    log.Debug("第一次访问页面");
            //    return true;
            //}
            //else
            //{
            //    bool flag = RefreshSign.Rows.IndexOf(RefreshSign.Rows.Find(GetCurSubmitTime())) > -1;//当前提交时间是否存在
            //    if (flag)
            //        log.Debug("提交时间存在");
            //    else
            //        log.Debug("无效的提交时间");
            //    return flag;
            //}
            #endregion

            HttpCookie cookie = GetRefreshTicket();
            string pageGuid = GetCurPageGuid();
            if (cookie.Values.Count > 0)
            {
                bool flag;
                if (cookie.Values[pageGuid] != null)
                    flag = cookie.Values[pageGuid].IndexOf(GetCurSubmitTime()) > -1;
                else
                    flag = true;//防止出现异常,总是可以提交
                if (flag)
                    log.Debug("提交时间存在,可以提交");
                else
                    log.Debug("无效的提交时间");
                return flag;
            }
            return true;
        }


        /// <summary>
        /// 得到已保存的提交时间,没有新建,有返回
        /// </summary>
        /// <returns></returns>
        private HttpCookie GetRefreshTicket()
        {
            #region Session模式,返回值改为DataTable

            //DataTable RefreshSession;
            //if (Session[REFRESH_TICKET_NAME] == null)
            //{
            //    RefreshSession = new DataTable("RefreshSession");
            //    DataColumn newColumn;

            //    newColumn = new DataColumn("submitTime", System.Type.GetType("System.String"));
            //    RefreshSession.Columns.Add(newColumn);

            //    DataColumn[] columnArray = new DataColumn[1];
            //    columnArray[0] = newColumn;
            //    RefreshSession.PrimaryKey = columnArray;


            //    newColumn = new DataColumn("GUID", System.Type.GetType("System.String"));
            //    RefreshSession.Columns.Add(newColumn);

            //    Session[REFRESH_TICKET_NAME] = RefreshSession;

            //    log.Debug("Session不存在,初始化");
            //}
            //else
            //{
            //    RefreshSession = Session[REFRESH_TICKET_NAME] as DataTable;

            //    log.Debug("读取已存在的Session,当前DataTable中存在的记录数为:" + RefreshSession.Rows.Count);
            //    foreach (DataRow row in RefreshSession.Rows)
            //    {
            //        log.Info("row['submitTime']:" + row["submitTime"] + "    row['GUID']:" + row["GUID"]);
            //    }             
            //}          
            //return RefreshSession;
            #endregion

            #region Cookie模式,返回值为Cookie

            HttpCookie cookie;
            if (Request.Cookies[REFRESH_TICKET_NAME] == null)
            {
                cookie = new HttpCookie(REFRESH_TICKET_NAME);
                Response.AppendCookie(cookie);
                log.Debug("Cookie不存在,初始化");
            }
            else
            {
                cookie = Request.Cookies[REFRESH_TICKET_NAME];

                log.Debug("读取已存在的Cookie,当前Cookie中存在的记录数为:" + cookie.Values.Count + "具体有如下几条:");

                for (int i = 0; i < cookie.Values.Count; i++)
                    log.Info("cookie[" + cookie.Values.GetKey(i) + "]:" + cookie.Values[i]);
            }
            return cookie;
            #endregion
        }


        /// <summary>
        /// 获取当前提交时间
        /// </summary>
        /// <returns></returns>
        private string GetCurSubmitTime()
        {
            string submitTime = Request.Params[HIDDEN_FIELD_NAME] == null ? "" : Request.Params[HIDDEN_FIELD_NAME].ToString();
            log.Debug("执行GetCurSubmitTime:submitTime为:" + submitTime);
            return submitTime;
        }


        /// <summary>
        /// 设置页面唯一标识,通过Guid标识来区分每个页面自己的提交时间
        /// </summary>
        private string SetCurPageGuid()
        {
            string guid;
            if (!IsPostBack)
            {
                if (Request.Params[HIDDEN_PAGE_GUID] == null)
                {
                    guid = System.Guid.NewGuid().ToString();
                    log.Debug("SetCurPageGuid注册了一个新的标识:" + guid);
                }
                else
                    guid = GetCurPageGuid();

            }
            else
            {
                guid = GetCurPageGuid();              
            }

            ClientScript.RegisterHiddenField(HIDDEN_PAGE_GUID, guid);
            return guid;
        }



        /// <summary>
        /// 得到当前页面的唯一标识
        /// </summary>
        /// <returns></returns>
        private string GetCurPageGuid()
        {
            string pageGuid = Request.Params[HIDDEN_PAGE_GUID] == null ? "none" : Request.Params[HIDDEN_PAGE_GUID].ToString();
            log.Debug("执行GetCurPageGuid()后Page_GUID为:" + pageGuid);
            return pageGuid;
        }

需要刷新判断功能时新页面只需继续该类就可,通过引用属性IsPageRefreshed识别"为真表示刷新,假则是正常提交",将数据库的操作写在
if(!IsPageRefreshed)
{
   数据库操作
}
即可,如果是刷新不会执行,代码中注释部分使用的是Session方式保存票证,因为session比较容易丢失且占内存,所以使用cookie,
分享到:
评论

相关推荐

    表单重复提交问题1

    要彻底解决表单重复提交的问题,有以下几种策略: - **禁用表单按钮**:在表单提交后,禁用提交按钮,防止用户点击多次。 - **使用POST/REDIRECT/GET模式**:在处理POST请求后,使用`sendRedirect()`重定向到一个新...

    asp.net防止刷新时重复提交(可禁用工具条刷新按钮)

    无论是通过自定义`LoadViewState`和`SaveViewState`来手动管理状态,还是使用RefreshModule这样的第三方控件,其核心目的都是为了解决页面刷新导致的重复提交问题,从而提升用户体验并保证数据的准确性。

    ThinkPHP 防止表单重复提交的方法

    总结来说,ThinkPHP提供了方便的表单token验证机制来防止表单重复提交,但在面对浏览器缓存问题时,需要通过修改框架的模板渲染行为来彻底解决问题。通过上述的详细步骤和解释,可以更深入理解在ThinkPHP框架中如何...

    测绘资质管理信息系统修改建议.doc

    - **信息重复检测**:在提交时自动检查人员和设备信息的唯一性,避免冲突。 - **附件管理**:修复已上传附件无法删除或显示问题,确保信息同步准确。 7. **错误提示与帮助文档**: - **错误处理**:识别并修复如...

    网趣网上购物系统时尚版V13.0

    应用户强烈要求,时尚版具有商品批量添加功能,可自定义一次性添加的商品个数,可一次提交保存所有商品信息,抛弃单调、重复的工作,网趣时尚版新版脱颖而出,让您的管理工作更轻松,管理更方便! 十、订单自动...

    各领域的主要功能测试

    7. **表单重复提交**: 验证表单在提交后能否阻止重复提交。 8. **重名冲突**: 在修改时避免输入与现有记录相同的名称。 9. **添加与修改一致性**: 确保添加和修改操作的要求保持一致。 10. **删除功能测试**: 测试...

    金碟软件快捷键.pdf

    - Ctrl+Shift+Delete:彻底删除选中的文本或文件,可能涉及到永久删除,不经过回收站。 2. 导航快捷键: - F1:通常用作帮助功能的快捷键,提供软件使用帮助信息。 - F3:用于打开或激活搜索功能。 - F4:在...

    asce-common:可以重用并轻松集成到任何项目中的android通用代码

    这些工具类能够帮助开发者解决日常开发中的常见问题,提高开发效率。 6. **事件总线**: 为了实现组件间的通信,ASCE-Common可能集成了EventBus或RxBus这样的事件总线框架,使得组件之间的解耦更加彻底,消息传递...

    网趣网上购物系统的部分特点与精华

    应用户强烈要求,时尚版具有商品批量添加功能,可自定义一次性添加的商品个数,可一次提交保存所有商品信息,抛弃单调、重复的工作,网趣时尚版新版脱颖而出,让您的管理工作更轻松,管理更方便! 十、订单自动...

    键盘快捷键

    比**Ctrl+F5**更为彻底,不仅刷新页面,还会清除与页面相关的缓存数据。 **Ctrl+Shift+F8**:选择模式。某些软件中提供的功能,允许用户通过连续按ESC键来取消选择模式。 **Ctrl+F9**:插入大括号 `{}`。在编程...

    测试培训教材

    5、QC9.0安装问题解决 - JBOSS启动不了 原因:8080端口被其他应用程序占用。 用netstat查看谁占用了8080端口 netstat -ano 解决办法1:修改其他程序的端口使用 解决办法2:修改JBOSS的启动端口 JBOSS_HOME\...

    7z网上购物

    特别提醒:如果后台添加或删除数据刷新后无变化,即为网站无写权限,请如下操作:找到本程序的目录,先在工具-&gt;文件夹选项-&gt;查看-&gt;去掉“使用简单文件共享”确定。然后右键点本程序目录所在目录-&gt;属性-&gt;安全-&gt;添加...

    50用户在线考试系统

    课程管理员删除的试卷仅仅是做了删除标记,只有最高管理员才能彻底删除试卷。 4、知识点管理 课程管理员可以设计各种难度和级别的知识点,方便给试题添加相关属性。 (二)、题库管理 1、题型管理 本考试系统一共...

Global site tag (gtag.js) - Google Analytics