Asp.Net防止多次提交数据(转)


Asp.Net防止多次提交数据


RefreshAction类:

public static class RefreshAction
{
   // 常量
   //服务端票证key
   
   public const string LastRefreshTicketEntry = "__LASTREFRESHTICKET"; //客户端票证key
   public const string CurrentRefreshTicketEntry = "__CURRENTREFRESHTICKET"; //用来保存是否是重复刷新的属性的key
   public const string PageRefreshEntry = "IsPageRefresh"; // 检测F5按钮是否被按下
   
   public static void Check(HttpContext ctx)
   {
      //初始化服务端票证
      EnsureRefreshTicket(ctx); //从Session里读取上一次提供的票证
      
      int lastTicket = GetLastRefreshTicket(ctx); //从请求里的隐藏域里读取当前页面的票证
      int thisTicket = GetCurrentRefreshTicket(ctx); // 对比两个票证
      if (thisTicket != lastTicket || (thisTicket == lastTicket && thisTicket == 0))
      {
         //如果当前的票证值大于上一次的票证值 或者
         //当前票证值等于上一次票证值,并且当前票证值为0(这是第一次刷新)
         //那么更新Session里上一次的票证值为当前票证值
         UpdateLastRefreshTicket(ctx, thisTicket); //设置当前页是否重复刷新属性为false
         ctx.Items[PageRefreshEntry] = false;
      }
      else
      {
         //设置当前页是否重复刷新属性为true;
         ctx.Items[PageRefreshEntry] = true;
      }
   }
   
   //确认上一次的票证不为空值
   private static void EnsureRefreshTicket(HttpContext ctx)
   {
      // Initialize the session slots for the page (Ticket) and the module (LastTicketServed)
      //初始化Session的最后一次票证值
      if (ctx.Session[LastRefreshTicketEntry] == null)
      ctx.Session[LastRefreshTicketEntry] = 0;
      
   }
   
   //从Session里得到上一次请求的票证值
   private static int GetLastRefreshTicket(HttpContext ctx)
   { //返回Session里保存的上一次票证值
   return Convert.ToInt32(ctx.Session[LastRefreshTicketEntry]);
}


//从当前请求里的到隐藏域里保存的当前票证值
private static int GetCurrentRefreshTicket(HttpContext ctx)
{
   return Convert.ToInt32(ctx.Request[CurrentRefreshTicketEntry]);
}

// 将当前的票证值保存到Session里的上一次刷新的票证值
private static void UpdateLastRefreshTicket(HttpContext ctx, int ticket)
{
   ctx.Session[LastRefreshTicketEntry] = ticket;
}
}

// 来源:www.CSFramework.com, C/S结构框架学习网




RefreshModule 类:

//Provides module initialization and disposal events to the implementing class
public class RefreshModule : IHttpModule
{
   //初始化模块
   public void Init(HttpApplication app)
   { // Register for pipeline events
   //注册请求关联状态时的事件处理器,就是说当一个请求到达服务器,
   //那么首先触发这个事件,由OnAcquireRqeustState事件处理
   app.AcquireRequestState = new EventHandler(this.OnAcquireRequestState);
}

// IHttpModule::Dispose
public void Dispose() { }

//判断是否是F5或前进/后退操作
private void OnAcquireRequestState(object sender, EventArgs e)
{
   //得到访问的HTTP上下文
   HttpApplication app = (HttpApplication)sender;
   HttpContext ctx = app.Context; //检查是否是F5操作
   if (ctx != null && ctx.Session != null)
   RefreshAction.Check(ctx);
   return;
}
}

// 来源:www.CSFramework.com, C/S结构框架学习网



定义一个基类,所有提交数据的页面继承:SubmitOncePage

public class SubmitOncePage : PageBaseRefreshCounter
{
   
   #region Constants
   // 常量
   public const string RefreshTicketCounter = "RefreshTicketCounter";
   #endregion
   
   // Ctor
   
   public SubmitOncePage()
   {
      // Register a PreRender handler
      //注册页面呈现前的事件处理器
      this.PreRender = new EventHandler(RefreshPage_PreRender);
      
   }
   
   //标志页面是否按F5进行重复刷新的标志属性
   public bool IsPageRefresh
   {
      get
      {
         object o = HttpContext.Current.Items[RefreshAction.PageRefreshEntry];
         if (o == null)
         return false;
         return (bool)o;
      }
      set
      {
         object o = HttpContext.Current.Items[RefreshAction.PageRefreshEntry];
         if (o != null)
         HttpContext.Current.Items[RefreshAction.PageRefreshEntry] = false;
      }
   }
   
   //增加刷新票证的内部计数器
   public void TrackRefreshState()
   { //初始化刷新计数器
   InitRefreshState();
   //将刷新计数器加1,然后放进Session
   int ticket = Convert.ToInt32(Session[RefreshTicketCounter]) 1;
   Session[RefreshTicketCounter] = ticket;
}


#region Private Members
// Create the hidden field to store the current request ticket
//创建隐藏域来保存当前请求的票证值

private void SaveRefreshState()
{
   //将票证计数器的值加1,然后将此值注册到当前票证隐藏域中
   int ticket = Convert.ToInt32(Session[RefreshTicketCounter]) 1;
   this.ClientScript.RegisterHiddenField(RefreshAction.CurrentRefreshTicketEntry, ticket.ToString());
}

//初始化刷新计数器
private void InitRefreshState()
{
   if (Session[RefreshTicketCounter] == null)
   Session[RefreshTicketCounter] = 0;
}

// PreRender事件处理器
private void RefreshPage_PreRender(object sender, EventArgs e)
{
   //在页面呈现之前就保存票证值到隐藏域
   SaveRefreshState();
}

#endregion

}

// 来源:www.CSFramework.com, C/S结构框架学习网



最后在Web.Config文件内定义一个<httpModules>


  <system.web>

 

    <httpModules>

      <add name="CSFramework_RefreshModule" type="CSFramework.BLL.RefreshModule"/>

    </httpModules>

 





 

版权声明:本文为开发框架文库发布内容,转载请附上原文出处连接
C/S框架网
上一篇:Asp.Net防止恶意刷新网页
下一篇:Asp.Net FileUpload类实现上传文件(C/S框架网开源)
评论列表

发表评论

评论内容
昵称:
关联文章

Asp.Net防止多次提交数据()
Asp.Net防止恶意刷新网页
Asp.Net SignalR介绍
什么是ASP.NET WebApi控制器(APIController)?
ASP.NET MVC快速入门(一)
使用dalBaseDataDict类提交单表数据
C#.NET 处理字符串数据去左右空格、全角半角
ASP.NET Web API入门介绍(一)
[原创]Asp.Net的GridView绑定空数据显示标题
WebApi接口安全机制:API接口限流防止恶意访问 ThrottlingHandler消息处理机制
HTTP-POST提交数据:ContentType=application/json 内容格式
ASP.NET MVC中几种常用的ActionResult详解
C# ASP.NET 实现文件断点续传
With语法实现SQL树结构数据查询()
无DAL数据访问层提交单个表的数据
DbDataUpdate - 自动提交对象模型数据 - 常用数据类型测试
VUE+Element-UI入门到精通:客户详情页资料编辑及提交数据
什么是Web Api? ASP.NET Web Api体系架构
ASP.NET Web Forms - HTML 页面
1.1 什么是ASP.NET / WebAPI / MVC?

热门标签
.NET5 .NET6 .NET7 APP Auth-软件授权注册系统 Axios B/S B/S开发框架 Bug Bug记录 C#加密解密 C#源码 C/S CHATGPT CMS系统 CodeGenerator CSFramework.DB CSFramework.EF CSFrameworkV1学习版 CSFrameworkV2标准版 CSFrameworkV3高级版 CSFrameworkV4企业版 CSFrameworkV5旗舰版 CSFrameworkV6.0 DAL数据访问层 Database datalock DbFramework Demo教学 Demo下载 DevExpress教程 DOM EF框架 Element-UI EntityFramework ERP ES6 Excel FastReport GIT HR IDatabase IIS JavaScript LINQ MES MiniFramework MIS NavBarControl Node.JS NPM OMS ORM PaaS POS Promise API Redis SAP SEO SQL SQLConnector TMS系统 Token令牌 VS2022 VSCode VUE WCF WebApi WebApi NETCore WebApi框架 WEB开发框架 Windows服务 Winform 开发框架 Winform 开发平台 WinFramework Workflow工作流 Workflow流程引擎 版本区别 报表 踩坑日记 操作手册 代码生成器 迭代开发记录 基础资料窗体 架构设计 角色权限 开发sce 开发技巧 开发教程 开发框架 开发平台 开发指南 客户案例 快速搭站系统 快速开发平台 秘钥 密钥 权限设计 软件报价 软件测试报告 软件简介 软件开发框架 软件开发平台 软件开发文档 软件体系架构 软件下载 软著证书 三层架构 设计模式 生成代码 实用小技巧 收钱音箱 数据锁 数据同步 微信小程序 未解决问题 文档下载 喜鹊ERP 喜鹊软件 系统对接 详细设计说明书 行政区域数据库 需求分析 疑难杂症 蝇量级框架 蝇量框架 用户管理 用户开发手册 用户控件 在线支付 纸箱ERP 智能语音收款机 自定义窗体 自定义组件 自动升级程序