随笔 - 394  文章 - 0  评论 - 946  阅读 - 143万 

上一篇主要讲解了如何利用ActionFilter过滤关键字,这篇主要讲解如何利用自己打造的ModelBinder来过滤关键字。

首先,我们还是利用上一篇中的实体类,但是我们需要加上DataType特性,以便于我们构造的ModelBinder通过DataTypeName识别出来:

   1:  using System.ComponentModel.DataAnnotations;
   2:  using System.Web.Mvc;
   3:   
   4:  namespace MvcApplication1.Models
   5:  {
   6:      public class TestModel
   7:      {
   8:          public int TID { get; set; }
   9:   
  10:          [DataType("TName")]
  11:          public string TName { get; set; }
  12:   
  13:          [DataType("TSite")]
  14:          public string TSite { get; set; }
  15:      }
  16:  }

然后我们新建一个FilterModelBinder的类,其中内容如下:

   1:  using System;
   2:  using System.Collections.Generic;
   3:  using System.Linq;
   4:  using System.Web;
   5:  using System.Web.Mvc;
   6:   
   7:  namespace MvcApplication1
   8:  {
   9:      public class FilterModelBinder:DefaultModelBinder
  10:      {
  11:          public override object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
  12:          {
  13:              var valueShouldFilter = bindingContext.ModelMetadata.DataTypeName;
  14:              if (valueShouldFilter == "TName" || valueShouldFilter == "TSite")
  15:              {
  16:                  var resultProvider = bindingContext.ValueProvider.GetValue(bindingContext.ModelName);
  17:                  if (resultProvider != null)
  18:                  {
  19:                      string result = resultProvider.AttemptedValue;
  20:                      result = result.Replace("<", "&lt;").Replace(">", "&gt;");
  21:                      return result;
  22:                  }
  23:              }
  24:   
  25:              return base.BindModel(controllerContext, bindingContext);
  26:          }
  27:      }
  28:  }

 

第13行,主要是获取我们需要验证的DataTypeName.

第15行,获取需要验证的值,然后替换,最后返回即可.

 

上面做完后,在Global.asax中,我们需要指定一下:

   1:  protected void Application_Start()
   2:          {
   3:              AreaRegistration.RegisterAllAreas();
   4:   
   5:              WebApiConfig.Register(GlobalConfiguration.Configuration);
   6:              FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);
   7:              RouteConfig.RegisterRoutes(RouteTable.Routes);
   8:              BundleConfig.RegisterBundles(BundleTable.Bundles);
   9:   
  10:              ModelBinders.Binders.DefaultBinder = new FilterModelBinder();
  11:          }

 

这样,我们就能使用我们自己的ModelBinder了,下面开始测试:

image

我们输入的内容如上图所示,当点击”添加”按钮的时候,确弹出如下的错误提示:

image

看来,系统会自动检测我们的输入值,发现有非法字符,会弹出错误提示,还好我们可以通过web.config配置一下,让其通过验证:

打开最外层的Web.config,输入以下节点:

   1:  <configuration>
   2:    <system.web>
   3:     <httpRuntime requestValidationMode="2.0" />
   4:    </system.web>
   5:    <pages validateRequest="false">
   6:    </pages>
   7:  </configuration>

然后保存,运行,我们看到,系统成功跑了起来,最后的结果如下:

image

我们可以看到,通过我们自定义的ModelBinder,系统自动将非法字符进行了替换,非常方便。

MVC中处处AOP,现在我们就可以利用现有的知识做一个全局过滤器了。是不是感觉很方便呢?

 

百度网盘源代码下载

微云源代码下载

 

2014.04.26更新

由于上面的例子不能体现其通用性,毕竟我们定义了一个TSite的datatype和一个TName的datatype。

其实我们完全可以给需要过滤的字段加上一个 [DataType("ShouldBeFilter")]的定义,这样无论是什么实体,只要包含了这个定义,都可以被过滤掉关键字了:

实体类定义:

1
2
3
4
5
6
7
8
9
10
public class TestModel
   {
       public int TID { get; set; }
 
       [DataType("ShouldBeFilter")]
       public string TName { get; set; }
 
       [DataType("ShouldBeFilter")]
       public string TSite { get; set; }
   }

 通用过滤方法:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
using System.Web.Mvc;
 
namespace MvcApplication1
{
    public class FilterModelBinder:DefaultModelBinder
    {
        public override object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
        {
            var valueShouldFilter = bindingContext.ModelMetadata.DataTypeName;
            if (valueShouldFilter == "ShouldBeFilter")
            {
                var resultProvider = bindingContext.ValueProvider.GetValue(bindingContext.ModelName);
                if (resultProvider != null)
                {
                    string result = resultProvider.AttemptedValue;
                    result = result.Replace("<", "<").Replace(">", ">");
                    return result;
                }
            }
 
            return base.BindModel(controllerContext, bindingContext);
        }
    }
}

 

posted on   程序诗人  阅读(418)  评论(0编辑  收藏  举报
编辑推荐:
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
阅读排行:
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 【自荐】一款简洁、开源的在线白板工具 Drawnix
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY
· Docker 太简单,K8s 太复杂?w7panel 让容器管理更轻松!
历史上的今天:
2011-04-24 HTTPModules与HTTPHandler【转】
点击右上角即可分享
微信分享提示