ASP.NET MVC & EF 构建智能查询 二、模型的设计“.NET研究”与ModelBinder

简介:   在第一篇中,我讲解了我们要做智能查询的原因,以及基本的解决方案设计。从这篇开始我们开始讲解它的实现过程。  其实在写这一系列文章之初,我其实是想由底至上去讲解,但是我又整理了一遍代码才发现,其实如果不了解最表面的东西,也是不太好深入的。

  在第一篇中,我讲解了我们要做智能查询的原因,以及基本的解决方案设计。从这篇开始我们开始讲解它的实现过程。

  其实在写这一系列文章之初,我其实是想由底至上去讲解,但是我又整理了一遍代码才发现,其实如果不了解最表面的东西,也是不太好深入的。

  所以我们的第二篇文章就来讲一下我们这个智能查询框架中最浅,但也是使用最频繁的部分,也就是Model。

  首先我们的Entity  或者说数据库的结构如下:

image  另外如下面代码,我们有一个用于传递name=value对,及查询谓词的model:

 
 
public ActionResult Index(QueryModel model)
{
using (var db = new DbEntities())
{
var list
= db.Users.Where(model).ToList();
return View(lis上海企业网站制作t);
}
}

  我命名之为QueryModel。它由Action的参数传入,再传入EF的Where扩展方法,它是构建Lambda表达式的原型,上面是我们的一个通用的查询Action的代码。

  而QueryModel的代码为:

image  QueryModel的唯一一个属性Items,是一个ConditionItem的集合,它里面存着所有的查询条件。而ConditionItem里面的属性:

  1. Field表示要查询的目标属性的名称,我们的设定它是支持子属性查询的,例如 “Profile.MyUser.Id”。
  2. Method则是当前使用的谓词,它是QueryMethod的一个枚举值。
  3. OrGroup是一个字符串,是一个OrGroup的多个表达式将会以Or操作符进行关联,然后再And。
  4. Prefix是一个分类的前缀,我们假定一个Action可能处理多个查询条件组的时候为了分开这些查询条件而加的属性。
  5. Value则是这个表达试的值。

  而我们在页面上类似:

 
   
< form action ="" method ="post" >
姓名:
< input id ="Name" name ="[Like]Name" type ="text" value ="" />
Email:
< input id ="Email" name ="[Equal]Email" type ="text" value ="" />< br />
Id:
< input id ="Id" name ="[Equal]Id" type ="text" value ="" />
生日:
上海徐汇企业网站制作 style="color: #0000ff;">< input 上海闵行企业网站设计与制作an style="color: #ff0000;">id ="Birthday" name ="[Equal]Birthday" type ="text" value ="" />< br />
< input type ="submit" value ="查询" />
</ form >

  这样的表单,我们提交到服务器端,我们要进行ASP.NET MVC 中IModelBinder的转换,要将querystring 或 postdata中的KeyValuePair转换为我们的ConditionItem。

  当然,我们除了谓词Method之外,还有Prefix以及OrGroup要从客户端提交过来所以我们就要制定一个规则,我们约定:

  1. 中括号[]中的为查询谓词
  2. 小括号()中的为前缀Prefix
  3. 大括号{}中的为OrGroup

  我们可以通过以下IModelBinder的实现将Request.QueryString或Request.Form中的值转换到QueryModel中:

 
   
public class SearchModelBinder : IModelBinder
{
public object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
{
var model
= (QueryModel)(bindingContext.Model ?? new QueryModel());
var dict
= controllerContext.HttpContext.Request.Params;
var keys
= dict.AllKeys.Where(c => c.StartsWith( " [ " )); // 我们认为只有[开头的为需要处理的
if (keys.Count() != 0 )
{
foreach (var key in keys)
{
if ( ! key.StartsWith( " [ " )) continue ;
var val
= dict[key];
// 处理无值的情况
if ( string .IsNullOrEmpty(val)) 上海闵行企业网站制作yle="color: #0000ff;">continue ;
AddSearchItem(model, key, val);
}
}
return model;
}

/// <summary>
/// 上海网站建设yle="color: #008000;"> 将一组key=value添加入QueryModel.Items
/// </summary>
/// <param name="model"> QueryModel </param>
/// <param name="key"> 当前项的HtmlName </param>
/// <param name="val"> 当前项的值 </param>
public static void AddSearchItem(QueryModel model, string key, string val)
{
string field = "" , prefix = "" , orGroup = "" , method = "" ;
var keywords
= key.Split( ' ] ' , ' ) ' , ' } ' );
// 将Html中的name分割为我们想要的几个部分
foreach (var keyword in keywords)
{
if (Char.IsLetterOrDigit(keyword[ 0 ])) field = keyword;
var last
= keyword.Substring( 1 );
if (keyword[ 0 ] == ' ( ' ) prefix = last;
if (keyword[ 0 ] == ' [ ' ) method = last;
if (keyword[ 0 ] == ' { ' ) orGroup = last;
}
if ( string .IsNullOrEmpty(method)) return ;
if ( ! string .IsNullOrEmpty(field))
{
var item
= new ConditionItem
{
Field
= field,
Value
= val.Trim(),
Prefix
= prefix,
OrGroup
= orGroup,
Method
= (QueryMethod) Enum.Parse( typeof (QueryMethod), method)
};
model.Items.Add(item);
}
}
}

  当然我们还要在Global.asax中添加:

 
 
ModelBinders.Binders.Add( typeof (QueryModel), new SearchModelBinder());

  这样我们就可以在Action中获取到相应的查询条件了:

image  我们可以看到,从表单提交过来的数据我们已经正确地存放在QueryModel中了。

目录
相关文章
|
4月前
|
SQL 程序员
分享一个 .NET 通过监听器拦截 EF 消息写日志的详细例子
分享一个 .NET 通过监听器拦截 EF 消息写日志的详细例子
|
3月前
|
开发框架 前端开发 .NET
VB.NET中如何利用ASP.NET进行Web开发
在VB.NET中利用ASP.NET进行Web开发是一个常见的做法,特别是在需要构建动态、交互式Web应用程序时。ASP.NET是一个由微软开发的开源Web应用程序框架,它允许开发者使用多种编程语言(包括VB.NET)来创建Web应用程序。
69 5
|
3月前
|
前端开发 安全 Java
技术进阶:使用Spring MVC构建适应未来的响应式Web应用
【9月更文挑战第2天】随着移动设备的普及,响应式设计至关重要。Spring MVC作为强大的Java Web框架,助力开发者创建适应多屏的应用。本文推荐使用Thymeleaf整合视图,通过简洁的HTML代码提高前端灵活性;采用`@ResponseBody`与`Callable`实现异步处理,优化应用响应速度;运用`@ControllerAdvice`统一异常管理,保持代码整洁;借助Jackson简化JSON处理;利用Spring Security增强安全性;并强调测试的重要性。遵循这些实践,将大幅提升开发效率和应用质量。
72 7
|
4月前
|
开发框架 .NET 数据库连接
闲话 Asp.Net Core 数据校验(三)EF Core 集成 FluentValidation 校验数据例子
闲话 Asp.Net Core 数据校验(三)EF Core 集成 FluentValidation 校验数据例子
|
4月前
|
开发框架 JSON .NET
ASP.NET Core 标识(Identity)框架系列(三):在 ASP.NET Core Web API 项目中使用标识(Identity)框架进行身份验证
ASP.NET Core 标识(Identity)框架系列(三):在 ASP.NET Core Web API 项目中使用标识(Identity)框架进行身份验证
|
4月前
|
开发框架 .NET 开发工具
【Azure 应用服务】App Service 的.NET Version选择为.NET6,是否可以同时支持运行ASP.NET V4.8的应用呢?
【Azure 应用服务】App Service 的.NET Version选择为.NET6,是否可以同时支持运行ASP.NET V4.8的应用呢?
|
4月前
|
存储 开发框架 .NET
ASP.NET Web Api 使用 EF 6,DateTime 字段如何取数据库服务器当前时间
ASP.NET Web Api 使用 EF 6,DateTime 字段如何取数据库服务器当前时间
|
4月前
|
程序员 数据库
分享 2 个 .NET EF 6 只更新某些字段的方法
分享 2 个 .NET EF 6 只更新某些字段的方法
115 0
|
4月前
|
数据库
分享一个 .NET EF 6 扩展 Where 的方法
分享一个 .NET EF 6 扩展 Where 的方法
|
4月前
|
开发框架 前端开发 算法
分享 .NET EF6 查询并返回树形结构数据的 2 个思路和具体实现方法
分享 .NET EF6 查询并返回树形结构数据的 2 个思路和具体实现方法
下一篇
DataWorks