利用 DynamicLinq 实现简单的动态表达式构建查询
平时使用 LINQ 进行一些简单的条件拼接查询一般都会这样操作:
public class SearchInputDto
{
public string ConditionA { get; set; }
public int? ConditionB { get; set; }
public string ConditionC { get; set; }
}
这里有三个条件,是前端传入的搜索条件,然后我们来编写一个查询语句:
public Task Search(SearchInputDto input)
{
var queryResult = _db.Where(z=>(input.ConditionA == null || z.Name == input.ConditionA)
&& (input.ConditionB == null || z.Number == input.ConditionB)
&& (input.ConditionC == null || z.Address == input.ConditionC));
// 执行其他操作...
return Task.FromResult(0);
}
因为我们前端传入的条件不是固定的,所以有可能会出现有的条件没有传入的情况,如果是 SQL 的动态拼接 SQL 就可以了,而 Linq 你肯定是没法动态拼接的,只有自己构建一个表达式树传入到 IQuerable<T>.Where(Expression<Func<T,bool>> expression) 里面进行查询。
纯手工构建表达式树也不是不可以,只是略微麻烦,而我们则可以借助 System.Linq.Dynamic.Core 来方便的实现动态查询语句拼接。
他的常规用法如下:
官方 WIKI 地址:https://github.com/StefH/System.Linq.Dynamic.Core/wiki/Dynamic-Expressions
var query = db.Customers
.Where("City == @0 and Orders.Count >= @1", "London", 10)
.OrderBy("CompanyName")
.Select("new(CompanyName as Name, Phone)");
既然是字符串那么就可以拼接,我们来做一下改造。
首先去 NuGet 当中搜索 System.Linq.Dynamic.Core 库,安装之后我们来重新编写之前的查询范例,首先我们来写一个构建器,用于构建我们的表达式树:
using Abp.Runtime.Caching;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Linq.Dynamic.Core;
using System.Linq.Expressions;
using System.Reflection;
using System.Text;
namespace Abp.Linq.Expressions
{
public class ExpressionBuilder<TEntity, TSearchDto>
{
// 其实这里也可以通过传入 params Expression<Func<TRelateEntity, object>>[] selectFields 来构建
public Expression<Func<TEntity, bool>> Build(string[] excludeFields, TSearchDto dto)
{
var parameters = GenerateParametersDictionary(excludeFields, dto);
StringBuilder sb = new StringBuilder();
var fieldNames = parameters.Keys.ToList();
// 动态拼接
for (int i = 0; i < fieldNames.Count; i++)
{
sb.Append(fieldNames[i]).Append($" == @{i}").Append(" && ");
}
var lambdaStr = sb.ToString();
lambdaStr = lambdaStr.Substring(0, lambdaStr.Length - " && ".Length);
// 构建表达式
return DynamicExpressionParser.ParseLambda<TEntity, bool>(new ParsingConfig(), false, lambdaStr, parameters.Values.ToArray());
}
// 构建参数/值键值对,如果参数值为 NULL 则不进行构建
private Dictionary<string, object> GenerateParametersDictionary(string[] excludeFields, TSearchDto dto)
{
var typeInfo = typeof(TSearchDto);
var properties = typeInfo.GetProperties();
var parameters = new Dictionary<string, object>();
foreach (var property in properties)
{
var propertyValue = property.GetValue(dto);
if (propertyValue == null) continue;
if (excludeFields == null) continue;
if (excludeFields.Contains(property.Name)) continue;
if (parameters.ContainsKey(property.Name)) continue;
parameters.Add(property.Name, propertyValue);
}
return parameters;
}
}
}
用法很简单,用刚才的代码作为一个例子:
public Task Search(SearchInputDto input)
{
var builder = new ExpressionBuilder<EntityA,SearchInputDto>();
var queryResult = _db.Where(builder.Build(null,input));
// 执行其他操作...
return Task.FromResult(0);
}
可以看到已经变得十分简洁,这里仅仅作为抛砖引玉,其实还有更多高级的用法,这里不再赘述。
利用 DynamicLinq 实现简单的动态表达式构建查询的更多相关文章
- 利用SpEL 表达式实现简单的动态分表查询
这里的动态分表查询并不是动态构造sql语句,而是利用SpEL操作同一结构的不同张表. 也可以参考Spring Data Jpa中的章节http://docs.spring.io/spring-data ...
- 利用python实现简单词频统计、构建词云
1.利用jieba分词,排除停用词stopword之后,对文章中的词进行词频统计,并用matplotlib进行直方图展示 # coding: utf-8 import codecs import ma ...
- 超简单的集成表达式树查询组件,Sy.ExpressionBuilder 使用说明
Sy.ExpressionBuilder是一套依赖于表达式树上的集成的查询组件.设计的初衷没别的,就为了少写代码,让查询业务可以变得更加模式化.目前可以从nuget 获取到该组件. 来到查询,查询实体 ...
- ORM动态表达式树查询
前言 接口获取参数后,创建返回值模型的条件表达式作为参数,传入使用依赖注入实例化后的业务层. 业务层创建返回值模型的IQUERY后,再使用参数条件表达式.最后进行延迟查询. 代码实现 参数模型Demo ...
- EntityFramework动态多条件查询与Lambda表达式树
在常规的信息系统中, 我们有需要动态多条件查询的情况, 例如UI上有多个选择项可供用户选择多条件查询数据. 那么在.net平台Entity Framework下, 我们用Lambd ...
- 记录Linq中lambda动态表达式的使用方式
项目中有的时候我们会用到动态表达式的方式去查询数据,这里简单记录下个人的使用方式,方便使用↓ //构建参数表达式 ParameterExpression parameter = Expression. ...
- Java | 在 Java 中执行动态表达式语句: 前中后缀、Ognl、SpEL、Groovy、Jexl3
在一些规则集或者工作流项目中,经常会遇到动态解析表达式并执行得出结果的功能. 规则引擎是一种嵌入在应用程序中的组件,它可以将业务规则从业务代码中剥离出来,使用预先定义好的语义规范来实现这些剥离出来的业 ...
- [原创]java WEB学习笔记105:Spring学习---AOP介绍,相关概念,使用AOP,利用 方法签名 编写 AspectJ 切入点表达式
本博客的目的:①总结自己的学习过程,相当于学习笔记 ②将自己的经验分享给大家,相互学习,互相交流,不可商用 内容难免出现问题,欢迎指正,交流,探讨,可以留言,也可以通过以下方式联系. 本人互联网技术爱 ...
- LINQ to SQL 运行时动态构建查询条件
在进行数据查询时,经常碰到需要动态构建查询条件.使用LINQ实现这个需求可能会比以前拼接SQL语句更麻烦一些.本文介绍了3种运行时动态构建查询条件的方法.本文中的例子最终实现的都是同一个功能,从Nor ...
随机推荐
- vue项目运行
共分为以下六步: 1.安装node.js2.安装cnpm3.安装vue-cli脚手架构建工具4.用vue-cli构建项目5.安装项目所需的依赖6.运行项目 第1步:从node.js官网下载node.j ...
- lllll
- vue的一些注意点
每个 Vue 实例都会代理其 data 对象里所有的属性. 注意只有这些被代理的属性是响应的.如果在实例创建之后添加新的属性到实例上,它不会触发视图更新. 除了 data 属性,Vue实例暴露了一些有 ...
- Asp.net Zero 应用实战-最初部署问题
此时用的是aspnet-zero-core-4.3.0 1.前两天vs2017刚刚最新升级了,打开后就报错,然后就根据提示把每个项目文件中添加了 <PropertyGroup> <E ...
- sql中varchar(n),nvarchar(n) 长度性能及所占空间分析
sql中varchar(n),nvarchar(n) 长度性能及所占空间分析 1.varchar(n),nvarchar(n) 中的n怎么解释: nvarchar(n)最多能存n个字符,不区分中英文. ...
- k8s的基本使用
一.kubectl的命令参数 1)kubectl 能使用的命令.即查看帮助 [root@k8s6 ~]# kubectl kubectl controls the Kubernetes cluster ...
- electron培训 ppt
- pychrom 中文版
http://jingyan.baidu.com/article/a378c960daf80eb328283033.html
- redis学习-string常用命令
keys * :查询所有的key值 set:为指定键设置对应的值 get:获取指定键的值 mset:一次传入多个键值对 mget:一次获取多个键的值 del:删除指定键 strlen:获取指定键值的长 ...
- (23)The surprising connection between brain injuries and crime
https://www.ted.com/talks/kim_gorgens_the_surprising_connection_between_brain_injuries_and_crime/tra ...