前言  

  不少公司用的是ADO.NET的访问方式,估计不少朋友对于sql的拼写真是没太多好感,在没有一个封装足够好的底层的项目(还需要考虑sql注入等问题),特别是经过许多人接手之后,代码那叫一个惨不忍睹,本文借助【通用查询设计思想】这篇文章的思想和基于ADO.NET进行通用查询设计。

  【通用查询设计思想】这篇文章是基于核心方法GenerateQueryExpression来生成表达式的,基于这个思想我们重新写个方法拼装成类似sql中的where条件

      /// <summary>
/// 生成查询条件
/// </summary>
/// <typeparam name="TEntity">要查询的实体类型</typeparam>
public static string GenerateQueryCriterion<TEntity>(this IQuery<TEntity> query) where TEntity : class
{
var criterion = " where 1 = 1 ";
if (query == null) return criterion; var condition = " and ";
var returnStr = string.Empty;
var queryType = query.GetType(); foreach (PropertyInfo property in queryType.GetProperties())
{
var value = property.GetValue(query);
if (null == value)
continue; if (value is string)
{
var str = ((string)value).Trim();
value = string.IsNullOrEmpty(str) ? null : str;
} //针对QueryMode特性获取我们指定要查询的路径
foreach (var attribute in property.GetAttributes<QueryModeAttribute>())
{
var propertyPath = attribute.PropertyPath;
if (propertyPath == null)
propertyPath = property.Name; var conditionStr = CreateConditionString(value, propertyPath, attribute.Compare);
if (string.IsNullOrWhiteSpace(conditionStr))
continue;
returnStr += condition + conditionStr;
}
} return criterion + returnStr;
} /// <summary>
/// 生成sql条件
/// </summary>
/// <param name="value"></param>
/// <param name="propertyPath"></param>
/// <param name="compare"></param>
/// <returns></returns>
private static string CreateConditionString(object value, string propertyPath, QueryCompare compare)
{
switch (compare)
{
case QueryCompare.Equal:
return CreateEqualString(propertyPath, value);
case QueryCompare.GreaterThanOrEqual:
return CreateGreaterThanOrEqualString(propertyPath, value);
case QueryCompare.LessThanOrEqual:
return CreateLessThanOrEqualString(propertyPath, value);
case QueryCompare.Like:
return CreateLikeString(propertyPath, value);
default:
return null;
}
} /// <summary>
/// 生成sql的等于条件
/// </summary>
/// <param name="propertyPath"></param>
/// <param name="value"></param>
/// <returns></returns>
private static string CreateEqualString(string propertyPath, object value)
{
if (value == null) return string.Empty; if (value is string)
{
return propertyPath + "='" + value + "'";
} if (value is bool)
{
if (value.ToString() == "False")
return propertyPath + "=0";
return propertyPath + "=1";
} return propertyPath + "=" + value;
}

  方法返回的就是我们常见的“ 1 = 1 and System='A' ” 这样的查询条件,其他的类似 大于,小于,like这些条件,相信以朋友们高超的智慧肯定不是什么难事:)

  

  为避免新朋友没看过【通用查询设计思想】这篇文章,个人把这次的变动的类罗列出来

  查询基类(保留表达式的方法,兼容同时使用Linq和ADO.NET的情况:))  

    public class Query<TEntity> : IQuery<TEntity> where TEntity: class
{
/// <summary>
/// 指定查询条件
/// </summary>
protected Expression<Func<TEntity, bool>> Predicate; /// <summary>
/// 创建一个新的 <see cref="Query{TEntity}"/>
/// </summary>
public Query()
{
} /// <summary>
/// 创建一个指定查询条件的<see cref="Query{TEntity}"/>
/// </summary>
/// <param name="predicate">指定的查询条件</param>
public Query(Expression<Func<TEntity, bool>> predicate)
{
Predicate = predicate;
} /// <summary>
/// 生成表达式
/// </summary>
/// <returns></returns>
public Expression<Func<TEntity, bool>> GenerateExpression()
{
return Predicate.And(this.GetQueryExpression());
} /// <summary>
/// 生成sql条件
/// </summary>
/// <returns></returns>
public string GenerateSqlCriterion()
{
return this.GenerateQueryCriterion();
}
}

  查询模式(保留PropertyPath给表达式,PropertyPath2不是数组,因为ADO.NET中应该不会涉及到导航属性

    public class QueryModeAttribute : Attribute
{
/// <summary>
/// 比较方式
/// </summary>
public QueryCompare Compare { get; set; } /// <summary>
/// 对应属性路径(Linq + Expression)
/// </summary>
public string[] PropertyPath { get; set; } /// <summary>
/// 对应属性路径(sql + ADO.NET)
/// </summary>
public string PropertyPath2 { get; set; } /// <summary>
/// 查询字段
/// </summary>
public QueryModeAttribute(params string[] propertyPath)
{
PropertyPath = propertyPath;
} /// <summary>
/// 查询字段
/// </summary>
public QueryModeAttribute(QueryCompare compare,params string[] propertyPath)
{
PropertyPath = propertyPath;
Compare = compare;
} /// <summary>
/// 查询字段
/// </summary>
public QueryModeAttribute(QueryCompare compare, string propertyPath2)
{
PropertyPath2 = propertyPath2;
Compare = compare;
}
}

  查询实体

    public class AccountQuery : PageQuery<Account>
{
/// <summary>
/// 姓名
/// </summary>
[Query(QueryCompare.Equal, nameof(Account.Name))]
public string Name { get; set; }
    
    //其他查询条件... }

  这样的话我们的之前的表达式写法成了下面这样

  public class AccountQuery : PageQuery<Account>
{
/// <summary>
/// 姓名
/// </summary>
[Query(QueryCompare.Equal, new[]{nameof(Account.Name)})]
public string Name { get; set; }
    
    //其他查询条件... }  

  来看一下我们完整的使用例子,很多情况下我们是需要分页和添加类似于Order By或者Group By这样的高级条件(本例子前端是使用layui,数据库是mysql)

      public PageResult<AccountDto> GetAccount(AccountQuery query)
{
var sql = @"select xx,xx,xx from user ";
var criterion = query.GenerateSqlCriterion(); sql += criterion; return return MySqlHelper.GetPageResult<AccountDto>(sql, null, query);
}   /// <summary>
/// 获取分页数据
/// </summary>
/// <typeparam name="T"></typeparam>
/// <param name="cmdText"></param>
/// <param name="cmdParms"></param>
/// <param name="pageInfo"></param>
/// <param name="groupBy"></param>
/// <returns></returns>
  public static PageResult<T> GetPageResult<T>(string cmdText,
MySqlParameter[] cmdParms, PageInfo pageInfo, string groupBy = "") where T : class, new()
{
var result = new PageResult<T>
{
PageSize = pageInfo.PageSize,
PageIndex = pageInfo.PageIndex
};
//获取总数
var sqlCount = $"SELECT COUNT(*) FROM ({cmdText}) TableCount ";
var objCount = ExecuteScalar(CommandType.Text, sqlCount, cmdParms);
result.TotalCount = Convert.ToInt32(objCount); string pagingSql; //获取分页数据
if (string.IsNullOrWhiteSpace(groupBy))
{
pagingSql =
$"{cmdText} ORDER BY {pageInfo.Field ?? "id"} {pageInfo.Order ?? "desc"} " +
$"LIMIT {pageInfo.PageIndex}, {pageInfo.PageSize}";
}
else
{
//Group by
pagingSql =
$"{cmdText} GROUP BY {groupBy} ORDER BY {pageInfo.Field ?? "id"} {pageInfo.Order ?? "desc"}" +
$"LIMIT {pageInfo.PageIndex}, {pageInfo.PageSize}";
} var dt = ExecuteDataTable(CommandType.Text, pagingSql, cmdParms);
if (dt != null)
{
result.Data = dt.ToList<T>();
}
return result;
}

  当然,这只是我们的一般查询情况,实际情况中我们会遇到更复杂的sql,基于本文的中心主题,不再深入讨论其他场景。

  让我知道如果你有更好的想法!

  

通用查询设计思想(2)- 基于ADO.Net的设计的更多相关文章

  1. 平台+插件软件设计思想及基于COM的原型实现

    引言:我们已经习惯于一个人独立进行软件开发,每个人都使用自己的风格进行程序设计,但随着工程项目变大或者是对时间要求比较紧时,就需要几个人,十几个人,甚至是上百个人协作进行软件开发与设计,这时一个比较棘 ...

  2. Mysql高手系列 - 第27篇:mysql如何确保数据不丢失的?我们借鉴这种设计思想实现热点账户高并发设计及跨库转账问题

    Mysql系列的目标是:通过这个系列从入门到全面掌握一个高级开发所需要的全部技能. 欢迎大家加我微信itsoku一起交流java.算法.数据库相关技术. 这是Mysql系列第27篇. 本篇文章我们先来 ...

  3. FPGA设计思想与技巧(转载)

    题记:这个笔记不是特权同学自己整理的,特权同学只是对这个笔记做了一下完善,也忘了是从那DOWNLOAD来的,首先对整理者表示感谢.这些知识点确实都很实用,这些设计思想或者也可以说是经验吧,是很值得每一 ...

  4. FPGA/CPLD设计思想与技巧

    本文讨论的四种常用FPGA/CPLD设计思想与技巧:乒乓操作.串并转换.流水线操作.数据接口同步化,都是FPGA/CPLD逻辑设计的内在规律的体现,合理地采用这些设计思想能在FPGA/CPLD设计工作 ...

  5. 【学习笔记】【Design idea】一、Java异常的设计思想、性能相关、笔记

    1.前言: 异常.本该是多么优雅的东西,然而,得全靠自己在零散的信息中汇集. 学习笔记保持更新. 2.教材(参考资料) 其他 ①受检异常与非受检异常:https://www.cnblogs.com/j ...

  6. 基于 CSP 的设计思想和 OOP 设计思想的异同

    LinkerLin Go语言推崇的CSP编程模型和设计思想,并没有引起很多Go开发者包括Go标准库作者的重视.标准库的很多设计保留了很浓的OOP的味道.本篇Blog想比较下从设计的角度看,CSP和OO ...

  7. dapi 基于Django的轻量级测试平台一 设计思想

    GitHub:https://github.com/yjlch1016/dapi 一.项目命名: dapi:即Django+API测试的缩写 二.设计思想: 模拟性能测试工具JMeter的思路, 实现 ...

  8. 使用Unity3D的设计思想实现一个简单的C#赛车游戏场景

    最近看了看一个C#游戏开发的公开课,在该公开课中使用面向对象思想与Unity3D游戏开发思想结合的方式,对一个简单的赛车游戏场景进行了实现.原本在C#中很方便地就可以完成的一个小场景,使用Unity3 ...

  9. 掌握 Cinder 的设计思想 - 每天5分钟玩转 OpenStack(46)

    上一节介绍了 Cinder 的架构,这节讨论 Cinder 个组件如何协同工作及其设计思想. 从 volume 创建流程看 cinder-* 子服务如何协同工作 对于 Cinder 学习来说,Volu ...

随机推荐

  1. Mac下安装git

    gti下载地址 https://git-scm.com/downloads 一步一步来就完事了.安装完以后执行 git version 查看是否更新到了该版本

  2. C++11中map的用法

    最全的c++map的用法 1. map最基本的构造函数:map<string ,int>mapstring; map<int,string >mapint;map<sri ...

  3. 【UOJ】67 新年的毒瘤 &【BZOJ】1123 BLO

    [UOJ 67] 题目链接: 传送门 题解: 第一眼很懵逼……这什么鬼. 思考什么点复合条件……(o(>﹏<)o 1.树,也就是说还剩n-2条边,等价于要删去一个度数为m-n+2的点. 2 ...

  4. csc.exe的环境变量设置

    csc.exe使用来编译*.cs文件的,但必须要在安装目录下使用.所以需要设置一下环境变量. C#的环境变量设置 1.“win+R” 打开运行窗口,并输入 “cmd”: 2.运行“set path=% ...

  5. java中的数组二分法

    数组二分法意在以较快的速度查找到某个值的下标位置. 二分法的核心思想:找到一个数组的中间位置值,判断某个数值是在这个中间值的左边还是右边,如果是左边,将中间位置之前进行二分,二分后,结束位置变为原始中 ...

  6. Eureka的功能特性及相关配置

    1.服务提供者1.1服务注册服务提供者启动时,会通过rest请求的方式将自己注册到Eureka Server上,同时带上了自身服务的一些元数据信息.Eureka Server接收到请求后,将元数据信息 ...

  7. 基于Java实现红黑树的基本操作

    首先,在阅读文章之前,我希望读者对二叉树有一定的了解,因为红黑树的本质就是一颗二叉树.所以本篇博客中不在将二叉树的增删查的基本操作了,需要了解的同学可以到我之前写的一篇关于二叉树基本操作的博客:htt ...

  8. 前端利用百度开发文档给的web服务接口实现对某个区域周边配套的检索

    最近项目需要实现地图功能,以便于实现对房源周边配套设施的检索.内容如下 其实百度官方有对应的api,但是对于一个网站来说这样的样式难免有些难看 这样的结果显然不能满足当下的需求,所以我决定利用官方给的 ...

  9. 新手教程:不写JS,在MIP页中实现异步加载数据

    从需求谈起:在 MIP 页中异步加载数据 MIP(移动网页加速器) 的 加速原理 除了靠谱的 MIP-Cache CDN 加速外,最值得一提的就是组件系统.所有 JS 交互都需要使用 MIP 组件实现 ...

  10. MIP 内容声明

    从搜索结果页点出的 MIP 页面,其页面上的任何内容(包括但不限于广告.在线咨询.统计等组件)均视为在原站点上的投放和使用. MIP (Mobile Instant Pages - 移动网页加速器), ...