超简单的集成表达式树查询组件,Sy.ExpressionBuilder 使用说明
Sy.ExpressionBuilder是一套依赖于表达式树上的集成的查询组件。设计的初衷没别的,就为了少写代码,让查询业务可以变得更加模式化。目前可以从nuget 获取到该组件。
来到查询,查询实体需要继承 QueryPageModel或者 QueryModel,从名字也基本可以看出来,一个用于分页,一个无分页,你可以根据自己需求选用哪个方式,如下我选了带分页的方式。
public partial class AllManagerDto:QueryPageModel
接下来这个查询实际就拥有了我们这个插件的大多数功能。
属性名称约束
为了方便处理各种属性类型,我做了一些属性名称的约定。
时间范围查询 =>属性名称 以 Start,End 结尾 ,生成条件为 >= 和<=。
数字范围查询 =>属性名称 以 Min,Max 结尾 ,生成条件为 >= 和<=。
字符串查询 => 名字需要和表字段一致,生成条件为 Contains。
编号查询必须是以Id结尾,不然如果编号为支付串的查询方式会以Contains形式查询。
例如:

/// <summary>
/// 租户编号
/// </summary>
public virtual int? TenantIdMin { get; set; }
/// <summary>
/// 租户编号
///</summary>
public virtual int? TenantIdMax { get; set; }
/// <summary>
/// 创建时间
/// </summary>
public virtual DateTime? CreateTimeStart { get; set; }
/// <summary>
/// 创建时间
///</summary>
public DateTime? CreateTimeEnd { get; set; }
/// <summary>
/// 创建人编号
/// </summary>
public virtual string? CreateUserId { get; set; }
特性约束
应用ConditionAttribute 特性,参数为 (字段名称,条件,查询方式) 目前我定义了常用的范围的约束。范围类型枚举

/// <summary>
/// 高级搜索条件
/// </summary>
[Description("高级搜索条件")]
public enum EnumCondition
{
/// <summary>
/// 包含
/// </summary>
[Description("包含")]
Contains = 0, /// <summary>
/// 等于
/// </summary>
[Description("等于")]
Equal = 1, /// <summary>
/// 大于等于
/// </summary>
[Description("大于等于")]
GtEqual = 2, /// <summary>
/// 大于
/// </summary>
[Description("大于")]
Gt = 3, /// <summary>
/// 小于等于
/// </summary>
[Description("小于等于")]
LtEqual = 4, /// <summary>
/// 小于
/// </summary>
[Description("小于")]
Lt = 5, /// <summary>
/// 不等于
/// </summary>
[Description("不等于")]
NotEqual = 6, /// <summary>
/// SQL(In函数)
/// </summary>
[Description("SQL In")]
In = 7, /// <summary>
/// 在什么之间
/// </summary>
[Description("在什么之间")]
Between = 8, /// <summary>
/// 不包含
/// </summary>
[Description("不包含")]
NotContain = 9, /// <summary>
/// 从尾部匹配
/// </summary>
[Description("从尾部匹配")]
EndsWith = 10, /// <summary>
/// 从头部匹配
/// </summary>
[Description("从头部匹配")]
StartsWith = 11, /// <summary>
/// 不在范围内
/// </summary>
[Description("不在范围内")]
NotIn = 12, /// <summary>
/// 空的
/// </summary>
[Description("空的")]
IsEmpty = 13, /// <summary>
/// 不为空的
/// </summary>
[Description("不为空的")]
IsNotEmpty = 14, /// <summary>
/// 非Null的
/// </summary>
[Description("非Null的")]
IsNotNull = 15, /// <summary>
/// Null的
/// </summary>
[Description("Null的")]
IsNull = 16, /// <summary>
/// IsNullOrWhiteSpace
/// </summary>
[Description("IsNullOrWhiteSpace")]
IsNullOrWhiteSpace = 17, /// <summary>
/// IsNotNullNorWhiteSpace
/// </summary>
[Description("IsNotNullNorWhiteSpace")]
IsNotNullNorWhiteSpace = 18, /// <summary>
///
/// </summary>
[Description("枚举")]
HasFlag = 19, }
当然我们该有我们的查询方式(且和或)的约束,这就放上来

/// <summary>
/// 当前条件所属类型
/// </summary>
[Description("当前条件所属类型")]
public enum EnumConditionType
{
/// <summary>
/// 并
/// </summary>
[Description("并")]
And = 0, /// <summary>
/// 或
/// </summary>
[Description("或")]
Or = 1
}
这样就构成了我们的查询约束,一般情况下,当前表字段查询的话我们只要属性名和表字段名一直即可,例如查询用户表的下UserName,如下即可
/// <summary>
/// 用户名称
/// </summary>
public string? UserName { get; set; }
如果有那种不想暴露字段在外部的,这时我们的特性才会说显示出用户,例如我还是要查询UserName,但是暴露给前端的名称确是Uname,因为特性中的属性名优先级会高于查询模型中的名称,那我们可如下处理
[Condition("UserName", EnumCondition.Contains,EnumConditionType.And)]
public string?Uname { get; set; }
又或者我们的某个字段需要包含一个集合的情况,我们可以如下实现
[Condition("UserId",EnumCondition.In,EnumConditionType.And)]
public string?UserIds{ get; set; }
导航属性单个查询,例如我在用户表,要根据角色名称查询,我们只要如下定义即可(导航属性名【角色表】+“.”+角色表下的角色名称,注意这个英文的 .,这个才是精髓)
[Condition("Role.RoleName", EnumCondition.Contains,EnumConditionType.And)]
public string? RoleName { get; set; }
导航属性集合查询,例如我在角色表,要查询有分配用户名字叫老王的所有角色,我们只要如下定义即可(导航属性名+“[”+角色表下的角色名称+"]",注意这个英文的 [], [] 表示这个是个集合)
[Condition("Users[UserName]", EnumCondition.Contains,EnumConditionType.And)] public string? UserName{ get; set; }
特别说明,该组件还支持位移枚举的查询,使用也超级简单,和一般属性几乎无差,查询模型中如下定义即可。
/// <summary>
/// 性别
/// </summary>
public EnumGender? Gender { get; set; }
不参与查询特性
如果我们有个别参数是作用于别的用途,不直接参数查询,或者目前该插件处理不了的,我们可以通过该特性排除,如下使用
/// <summary>
/// 不参与查询
/// </summary>
[NotQuery]
public string TreeId { get; set; }
额外扩展
对于排序,有些情况要做到用户点击表头,然后由后端进行排序后返回,这里我也预留了空间,如下(input 为继承了:QueryPageModel 的模型),第一个参数表示 要排序的字段名,第二个参数true 表示倒序,false 表示正序。
input.AddOrderByItem(nameof(News.Id),true);
我还加了一个相对显得鸡肋的默认排序,如果前端有传排序过来的话,这个是无效的,使用方式如下:
input.DefaultOrderBy(nameof(News.CreateTime)
当我们了解了以上约定,我们定义一个相对完整的查询模型,如下

/// <summary>
/// 查询参数实体
/// </summary> public class AllManagerDto : QueryPageModel
{
/// <summary>
/// 创建时间 开始(时间必须以Start结尾)
/// </summary>
public DateTime? CreateTimeStart { get; set; } /// <summary>
/// 创建时间 结束(结束时间必须以End结尾)
/// </summary>
public DateTime? CreateTimeEnd { get; set; } /// <summary>
/// 角色编号
/// </summary>
[Condition("Role.Id", EnumCondition.In)]
public string RoleId { get; set; } /// <summary>
/// 角色名称
/// </summary> [Condition("Role.RoleName", EnumCondition.Contains)] public string RoleName { get; set; } /// <summary>
/// 系统名称
/// </summary> [Condition("Sys[SysName]", EnumCondition.Contains)] public string SysName { get; set; } /// <summary>
/// 名称
/// </summary>
public string UserName { get; set; } /// <summary>
/// 性别
/// </summary> public EnumGender? Gender { get; set; } [NotMapped]
public string TreeId { get; set; } /// <summary>
/// 年龄 开始(必须以Min结尾)
/// </summary>
public int? AgeMin { get; set; } /// <summary>
/// 年龄 结尾(必须以Max结尾)
/// </summary>
public int? AgeMax{ get; set; } }
现在我们就可以进行查询了,如下(留意这句,input.ToQueryModel<Manager>() 主要是这句把查询参数转换成表达式了)
/// <summary>
/// 查询
/// </summary>
/// <param name="input"></param>
/// <returns></returns>
public static List<Manager> GetAll(AllManagerDto input)
{
var list = GetList();
input.RoleId = "1";
input.Tel = "18888888888";
input.UserName = "张三";
input.Gender = EnumGender.Man;
input.CreateTimeStart = DateTime.Parse("2021-9-22");
var query = input.ToQueryModel<Manager>();
return list.AsQueryable().Where(query).ToList();
}
放个以前的效果图
好了,到这你又可以去撸代码了。
超简单的集成表达式树查询组件,Sy.ExpressionBuilder 使用说明的更多相关文章
- 将简单的lambda表达式树转为对应的sqlwhere条件
1.Lambda的介绍 园中已经有很多关于lambda的介绍了.简单来讲就是vs编译器给我带来的语法糖,本质来讲还是匿名函数.在开发中,lambda给我们带来了很多的简便.关于lambda的演变过程可 ...
- 利用 DynamicLinq 实现简单的动态表达式构建查询
平时使用 LINQ 进行一些简单的条件拼接查询一般都会这样操作: public class SearchInputDto { public string ConditionA { get; set; ...
- ORM动态表达式树查询
前言 接口获取参数后,创建返回值模型的条件表达式作为参数,传入使用依赖注入实例化后的业务层. 业务层创建返回值模型的IQUERY后,再使用参数条件表达式.最后进行延迟查询. 代码实现 参数模型Demo ...
- C#编程(六十六)----------表达式树总结
表达式树总结 基础 表达式树提供了一个将可执行代码转换成数据的方法.如果你要在执行代码之前修改或转换此代码,那么它是很有用的.有其是当你要将C#代码----如LINQ查询表达式转换成其他代码在另一个程 ...
- C#高级编程六十六天----表达式树总结【转】
https://blog.csdn.net/shanyongxu/article/details/47257139 表达式树总结 基础 表达式树提供了一个将可执行代码转换成数据的方法.如果你要在执行代 ...
- 转载:C#特性-表达式树
原文地址:http://www.cnblogs.com/tianfan/ 表达式树基础 刚接触LINQ的人往往觉得表达式树很不容易理解.通过这篇文章我希望大家看到它其实并不像想象中那么难.您只要有普通 ...
- C#特性-表达式树
表达式树ExpressionTree 表达式树基础 转载需注明出处:http://www.cnblogs.com/tianfan/ 刚接触LINQ的人往往觉得表达式树很不容易理解.通过这篇文章我希 ...
- 表达式树ExpressionTree
表达式树基础 转载需注明出处:http://www.cnblogs.com/tianfan/ 刚接触LINQ的人往往觉得表达式树很不容易理解.通过这篇文章我希望大家看到它其实并不像想象中那么难.您只要 ...
- asp.net core 排序过滤分页组件:sieve(2)表达式树的复习
在Sieve组件中使用了很多关于表达式树的知识,但在我们日常的工作中写表达式树的机会是非常少的,至少在我的编程生涯中没怎么写过表达式树(可能也就是3,4次).所以,为了能够看懂Sieve里面的源代码, ...
随机推荐
- K8s配置配置存活、就绪和启动探测器
kubelet 使用存活探测器来知道什么时候要重启容器. 例如,存活探测器可以捕捉到死锁(应用程序在运行,但是无法继续执行后面的步骤). 这样的情况下重启容器有助于让应用程序在有问题的情况下更可用. ...
- Java项目开发中实现分页的三种方式一篇包会
前言 Java项目开发中经常要用到分页功能,现在普遍使用SpringBoot进行快速开发,而数据层主要整合SpringDataJPA和MyBatis两种框架,这两种框架都提供了相应的分页工具,使用 ...
- AtCoder AGC003 简要题解
A 首先横向和纵向互相独立,因此只考虑横向的情况. 那么显然只要不只往一边走都一定存在一种构造方式,直接判断即可,复杂度 \(\mathcal{O}(|S|)\). B 首先相邻两个数同时配对两次可以 ...
- java实现HTTPS单向认证&TLS指定加密套件
1.HTTPS介绍 由于HTTP是明文传输,会造成安全隐患,所以在一些特定场景中,必须使用HTTPS协议,简单来说HTTPS=HTTP+SSL/TLS.服务端和客户端的信息传输都是通过TLS进行加密. ...
- 【转载】Locust实现集合点
直接编写接口事务脚本对后台接口进行测试:有时测试需要让所有并发用户完成初始化后再进行压力测试,这就需要类似于LoadRunner中的集合点的概念,由于框架本身没有直接封装,有如下办法实现: from ...
- C语言中的单引号和双引号的区别
首先肯定地说,二者是有区别的,不是说用谁都一样. 1.实质区别,代表的含义不同 'A'代表的是一个整数,而且这个整数对应的是编译器所采用的字符集中的字符序列对应的数值.所以'A'跟ASCII中的65意 ...
- 《PHP程序员面试笔试宝典》——如何巧妙地回答面试官的问题?
如何巧妙地回答面试官的问题? 本文摘自<PHP程序员面试笔试宝典> 所谓"来者不善,善者不来",程序员面试中,求职者不可避免地需要回答面试官各种"刁钻&quo ...
- 微服务从代码到k8s部署应有尽有系列(二、网关)
我们用一个系列来讲解从需求到上线.从代码到k8s部署.从日志到监控等各个方面的微服务完整实践. 整个项目使用了go-zero开发的微服务,基本包含了go-zero以及相关go-zero作者开发的一些中 ...
- Windows RestartManeger重启管理器
介绍 重启管理器API可以消除或是减少在完成安装或是更新的过程中系统需要重启的次数.软件安装或是更新过程之所以需要重启系统的原因在于一些需要更新的文件正在被运行中的程序或服务使用.而重启管理器可以 ...
- SonarQube之采购选型参考
SonarQube是DevOps实践中主流的一款质量内建工具,过插件机制,Sonar 可以集成不同的测试工具,代码分析工具,以及持续集成工具,比如pmd-cpd.checkstyle.findbugs ...