C# 动态构建表达式树(二)——构建 Select 和 GroupBy 的表达式
C# 动态构建表达式树(二)——构建 Select 和 GroupBy 的表达式
前言
在上篇中写了表达式的基本使用,为 Where 方法动态构建了表达式。在这篇中会写如何为 Select 和 GroupBy 动态构建(可以理解为动态表达式的其它常见形式)。
本文的操作方式似乎在实际使用中作用甚微,仅作为了解即可
准备工作
环境:.NET Framework 4.5,SQLServer 2017
建表脚本如下(由 SSMS 导出):
USE [default]
GO
/****** Object: Table [dbo].[Person] Script Date: 2021/6/9 12:06:43 ******/
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
CREATE TABLE [dbo].[Person](
[Id] [varchar](100) NOT NULL,
[Name] [nvarchar](50) NOT NULL,
[Age] [int] NOT NULL,
[Gender] [nvarchar](5) NOT NULL,
[Point] [int] NOT NULL,
[CreateTime] [datetime] NOT NULL,
CONSTRAINT [PK_Person] PRIMARY KEY CLUSTERED
(
[Id] ASC
)WITH (PAD_INDEX = OFF, STATISTICS_NORECOMPUTE = OFF, IGNORE_DUP_KEY = OFF, ALLOW_ROW_LOCKS = ON, ALLOW_PAGE_LOCKS = ON) ON [PRIMARY]
) ON [PRIMARY]
GO

为 Select 方法动态构建表达式
假设我们要查出 Person 表中的 Name、Age、Gender 字段,返回类型为 List<PersonResult> 的对象:
class PersonResult
{
public string Name { get; set; }
public int Age { get; set; }
public string Gender { get; set; }
}
常规写法:
List<PersonResult> personList
= context.Person.Select(p => new PersonResult
{
Name = p.Name,
Age = p.Age,
Gender = p.Gender
}).ToList();
动态组合写法:
ParameterExpression pe = Expression.Parameter(typeof(Person), "p"); # 创建形参 p
MemberExpression meName = Expression.MakeMemberAccess(pe, typeof(Person).GetProperty("Name")); # 要使用 MakeMemberAccess 方法
MemberExpression meAge = Expression.MakeMemberAccess(pe, typeof(Person).GetProperty("Age"));
MemberExpression meGender = Expression.MakeMemberAccess(pe, typeof(Person).GetProperty("Gender"));
Type personResultType = typeof(PersonResult);
MemberAssignment maName = Expression.Bind(personResultType.GetProperty("Name"), meName); # 使用 Bind 方法将目标类型的属性与源类型的属性值绑定
MemberAssignment maAge = Expression.Bind(personResultType.GetProperty("Age"), meAge);
MemberAssignment maGender = Expression.Bind(personResultType.GetProperty("Gender"), meGender);
NewExpression ne = Expression.New(personResultType); # 相当于 new 关键字创建一个对象
MemberInitExpression mie = Expression.MemberInit(ne, maName, maAge, maGender); # 相当于初始化时赋值操作
Expression<Func<Person, PersonResult>> personSelectExpression = Expression.Lambda<Func<Person, PersonResult>>(mie, pe);
var personList1 = context.Person.Select(personSelectExpression).ToList();
与构建 Where 方法的表达式差不多,主要是创建新对象以及赋值的写法需要注意。
为 GroupBy 方法动态构建表达式
假设我们要统计出 Person 表中的男生女生数量,返回类型为 List<PersonGroupByResult> 对象
class PersonGroupByResult
{
public string Gender { get; set; }
public int Count { get; set; }
}
常规写法:
List<PersonGroupByResult> personList =
context.Person.GroupBy(p => p.Gender)
.Select(p => new PersonGroupByResult
{
Gender = p.Key,
Count = p.Count()
}).ToList();
动态组合写法:
// 动态创建 GroupBy 中的 Expression
ParameterExpression pe = Expression.Parameter(typeof(Person), "p");
MemberExpression meGender = Expression.Property(pe, "Gender");
Expression<Func<Person, string>> groupByExpression = Expression.Lambda<Func<Person, string>>(meGender, pe);
// 动态创建 Select 中的 Expression
Type groupType = typeof(IGrouping<string, Person>); # 注意 GroupBy 函数返回的类型
ParameterExpression pge = Expression.Parameter(groupType, "pg");
MemberExpression meKeyGender = Expression.MakeMemberAccess(pge, groupType.GetProperty("Key")); # 获取其中的属性,与上面动态拼接 Select 相同
Type groupByResultType = typeof(PersonGroupByResult);
MemberAssignment maGender = Expression.Bind(groupByResultType.GetProperty("Gender"), meKeyGender); # 使用 Bind 方法将目标类型的属性与源类型的属性值绑定,与上面动态拼接 Select 相同
MethodInfo countMethod = typeof(Enumerable).GetMethods().Where(a => a.Name == "Count" && a.GetParameters().Length == 1)
.FirstOrDefault().MakeGenericMethod(typeof(Person)); # 获取 Count 方法
MemberAssignment maCount = Expression.Bind(groupByResultType.GetProperty("Count"), Expression.Call(countMethod, pge)); #使用 Bind 方法将目标类型的属性与源类型调用方法的返回值绑定
NewExpression ne = Expression.New(groupByResultType);
MemberInitExpression mie = Expression.MemberInit(ne, maGender, maCount);
Expression<Func<IGrouping<string, Person>, PersonGroupByResult>> personSelectExpression =
Expression.Lambda<Func<IGrouping<string, Person>, PersonGroupByResult>>(mie, pge);
var personList1 = context.Person.GroupBy(groupByExpression).Select(personSelectExpression).ToList();
需要注意的是查找 Count 方法的过程。通过查看定义发现,IGrouping 类型中并没有 Count 方法,而 IGrouping 实现了 IEnumerable,因此想到获取 Enumerable 这个 IEnumerable 实现类中的 Count 方法。

而 Enumerable中 的 Count 方法定义如下:

在查阅资料和多次尝试后,仍然无法直接获取到 Count(当仅传入方法名称时,提示有多个定义;当传入方法名称和参数时,一直返回为 null)。现通过参数个数来筛选,得到想要的方法。还需要注意的是,Count 方法为泛型方法,得到后还需要执行 MakeGenericMethod 以传入泛型类型。
通过查看 ChangeTracer 和 SQL 执行情况,发现即使我们使用的是 Enumerable 类型,依然是只返回了我们想要的结果,没有全表查询。这其中的奥秘还需要探索啊。
其它的一点思考
之前在工作中为了方便经常使用 Select 查询出匿名类,能否使用动态创建表达式的方式创建匿名类呢(我能想到的一种使用场景是,根据某些条件返回不同的字段,但这其实可以通过冗余字段实现)。在进行了很多尝试后,发现只能先写好一个匿名类,再 Select 这个匿名类的相关字段。虽然看似达到了目的,但不符合我们动态组合的要求,因此是没有意义的。
参考
c# – 使用反射创建lambda表达式,如x => new {..}
referencing desired overloaded generic method
How to create LINQ Expression Tree to select an anonymous type
How to use Expression to build an Anonymous Type?
C# 动态构建表达式树(二)——构建 Select 和 GroupBy 的表达式的更多相关文章
- C#动态构建表达式树(三)——表达式的组合
C#动态构建表达式树(三)--表达式的组合 前言 在筛选数据的过程中,可能会有这样的情况:有一些查询条件是公共的,但是根据具体的传入参数可能需要再额外增加一个条件.对于这种问题一般有两种方法: a. ...
- Lambda表达式树构建(上)
概述 Lambda是C#常用的语句,采用委托等方式,来封装真实的代码块.Lambda其实就是语法糖,是一个匿名函数,是一种高效的类似于函数式编程的表达式,Lambda简化了开发中需要编写的代码量.它可 ...
- c#——表达式树在LINQ动态查询
一般如果逻辑比较简单,只是存在有的情况多一个查询条件,有的情况不需要添加该查询条件 简单方式这样操作就可以了 public IQueryable<FileImport> DynamicCh ...
- [.net 面向对象程序设计进阶] (6) Lamda表达式(二) 表达式树快速入门
[.net 面向对象程序设计进阶] (6) Lamda表达式(二) 表达式树快速入门 本节导读: 认识表达式树(Expression Tree),学习使用Lambda创建表达式树,解析表达式树. 学习 ...
- C# 表达式树遍历(二)
一.前言 上一篇我们对表达式树有了初步的认识,这里我们将对表达式树进行遍历,只有弄清楚了他的运行原理,我们才可以对他进行定制化修改. 表达式系列目录 C# 表达式树讲解(一) C# 表达式树遍历(二) ...
- WebKit Inside: DOM树的构建
当客户端App主进程创建WKWebView对象时,会创建另外两个子进程:渲染进程与网络进程.主进程WKWebView发起请求时,先将请求转发给渲染进程,渲染进程再转发给网络进程,网络进程请求服务器.如 ...
- 【C#表达式树 开篇】 Expression Tree - 动态语言
.NET 3.5中新增的表达式树(Expression Tree)特性,第一次在.NET平台中引入了"逻辑即数据"的概念.也就是说,我们可以在代码里使用高级语言的形式编写一段逻辑, ...
- LinQ实战学习笔记(三) 序列,查询操作符,查询表达式,表达式树
序列 延迟查询执行 查询操作符 查询表达式 表达式树 (一) 序列 先上一段代码, 这段代码使用扩展方法实现下面的要求: 取进程列表,进行过滤(取大于10M的进程) 列表进行排序(按内存占用) 只保留 ...
- 不可不知的表达式树(1)Expression初探
说起Lambda表达式,大家基本都很熟悉了,而表达式树(Expression Trees),则属于80%的工作中往往都用不到的那种技术,所以即便不是什么新技术,很多人对其理解都并不透彻.此文意图从表达 ...
随机推荐
- 两万字长文,彻底搞懂Kafka!
1.为什么有消息系统 1.解耦合 2.异步处理 例如电商平台,秒杀活动. 一般流程会分为: 风险控制 库存锁定 生成订单 短信通知 更新数据 通过消息系统将秒杀活动业务拆分开,将不急需处理的业务放在后 ...
- 三、Linux部署MinIO分布式集群
MinIO的官方网站非常详细,以下只是本人学习过程的整理 一.MinIO的基本概念 二.Windows安装与简单使用MinIO 三.Linux部署MinIO分布式集群 四.C#简单操作MinIO 一. ...
- js继承方式及特征
1. 原型链继承 (原型链) function Parent() { this.fruits = ['apple', 'orange']; } Parent.prototype.sayHello = ...
- Woc,考场(面试)忘记打平衡树怎么办,Trie救你命
Woc,考场(面试)忘记打平衡树怎么办,Trie救你命 文本只发布于博客园,其他地方出现本文均是未经许可的盗版. 算法导入 众所周知平衡树很难打(大佬除外),还老是调错.万一这种事情发生在关键时刻你就 ...
- DVWA-全等级文件包含
DVWA简介 DVWA(Damn Vulnerable Web Application)是一个用来进行安全脆弱性鉴定的PHP/MySQL Web应用,旨在为安全专业人员测试自己的专业技能和工具提供合法 ...
- stm32 connot enter debug mode
dap 可以发现设备,stlink jlink 均无法发现设备,但是都不能下载.connot enter debug mode ,发现是vdda 未连接
- 单片机学习(八)点阵LED的使用
目录 一.点阵LED的理论知识 1. 点阵屏的打开方式 2. LED点阵简介 3. 点阵LED的结构和操作方式 4. 74HC595模块 PPT上的简介 串行转并行的工作原理 二.编码实现 1. C5 ...
- NOIP 模拟 $22\; \rm e$
题解 对于这个 \(abs\) 就是求大于 \(r\) 的最小值,小于 \(r\) 的最大值,建权值线段树或平衡树. 因为是 \(k\) 个点的联通块,就是求它们的 \(lca\) 到它们的链,可持久 ...
- mpu6050控制舵机云台
准备材料:2个舵机mg90,云台支架,1个arduino-uno,mpu6050 编程工具:VScode-platformio or ArduinoIDE 我使用的是VScode在编辑代码上会更方 ...
- 数据结构与算法-排序(十)桶排序(Bucket Sort)
摘要 桶排序和基数排序类似,相当于基数排序的另外一种逻辑.它是将取值范围当做创建桶的数量,桶的长度就是序列的大小.通过处理比较元素的数值,把元素放在桶的特定位置,然后遍历桶,就可以得到有序的序列. 逻 ...