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

Expression表达式目录树动态拼接 反射获取泛型方法

How to create LINQ Expression Tree to select an anonymous type

How to use Expression to build an Anonymous Type?

C# 动态构建表达式树(二)——构建 Select 和 GroupBy 的表达式的更多相关文章

  1. C#动态构建表达式树(三)——表达式的组合

    C#动态构建表达式树(三)--表达式的组合 前言 在筛选数据的过程中,可能会有这样的情况:有一些查询条件是公共的,但是根据具体的传入参数可能需要再额外增加一个条件.对于这种问题一般有两种方法: a. ...

  2. Lambda表达式树构建(上)

    概述 Lambda是C#常用的语句,采用委托等方式,来封装真实的代码块.Lambda其实就是语法糖,是一个匿名函数,是一种高效的类似于函数式编程的表达式,Lambda简化了开发中需要编写的代码量.它可 ...

  3. c#——表达式树在LINQ动态查询

    一般如果逻辑比较简单,只是存在有的情况多一个查询条件,有的情况不需要添加该查询条件 简单方式这样操作就可以了 public IQueryable<FileImport> DynamicCh ...

  4. [.net 面向对象程序设计进阶] (6) Lamda表达式(二) 表达式树快速入门

    [.net 面向对象程序设计进阶] (6) Lamda表达式(二) 表达式树快速入门 本节导读: 认识表达式树(Expression Tree),学习使用Lambda创建表达式树,解析表达式树. 学习 ...

  5. C# 表达式树遍历(二)

    一.前言 上一篇我们对表达式树有了初步的认识,这里我们将对表达式树进行遍历,只有弄清楚了他的运行原理,我们才可以对他进行定制化修改. 表达式系列目录 C# 表达式树讲解(一) C# 表达式树遍历(二) ...

  6. WebKit Inside: DOM树的构建

    当客户端App主进程创建WKWebView对象时,会创建另外两个子进程:渲染进程与网络进程.主进程WKWebView发起请求时,先将请求转发给渲染进程,渲染进程再转发给网络进程,网络进程请求服务器.如 ...

  7. 【C#表达式树 开篇】 Expression Tree - 动态语言

    .NET 3.5中新增的表达式树(Expression Tree)特性,第一次在.NET平台中引入了"逻辑即数据"的概念.也就是说,我们可以在代码里使用高级语言的形式编写一段逻辑, ...

  8. LinQ实战学习笔记(三) 序列,查询操作符,查询表达式,表达式树

    序列 延迟查询执行 查询操作符 查询表达式 表达式树 (一) 序列 先上一段代码, 这段代码使用扩展方法实现下面的要求: 取进程列表,进行过滤(取大于10M的进程) 列表进行排序(按内存占用) 只保留 ...

  9. 不可不知的表达式树(1)Expression初探

    说起Lambda表达式,大家基本都很熟悉了,而表达式树(Expression Trees),则属于80%的工作中往往都用不到的那种技术,所以即便不是什么新技术,很多人对其理解都并不透彻.此文意图从表达 ...

随机推荐

  1. 两万字长文,彻底搞懂Kafka!

    1.为什么有消息系统 1.解耦合 2.异步处理 例如电商平台,秒杀活动. 一般流程会分为: 风险控制 库存锁定 生成订单 短信通知 更新数据 通过消息系统将秒杀活动业务拆分开,将不急需处理的业务放在后 ...

  2. 三、Linux部署MinIO分布式集群

    MinIO的官方网站非常详细,以下只是本人学习过程的整理 一.MinIO的基本概念 二.Windows安装与简单使用MinIO 三.Linux部署MinIO分布式集群 四.C#简单操作MinIO 一. ...

  3. js继承方式及特征

    1. 原型链继承 (原型链) function Parent() { this.fruits = ['apple', 'orange']; } Parent.prototype.sayHello = ...

  4. Woc,考场(面试)忘记打平衡树怎么办,Trie救你命

    Woc,考场(面试)忘记打平衡树怎么办,Trie救你命 文本只发布于博客园,其他地方出现本文均是未经许可的盗版. 算法导入 众所周知平衡树很难打(大佬除外),还老是调错.万一这种事情发生在关键时刻你就 ...

  5. DVWA-全等级文件包含

    DVWA简介 DVWA(Damn Vulnerable Web Application)是一个用来进行安全脆弱性鉴定的PHP/MySQL Web应用,旨在为安全专业人员测试自己的专业技能和工具提供合法 ...

  6. stm32 connot enter debug mode

    dap 可以发现设备,stlink jlink 均无法发现设备,但是都不能下载.connot enter debug mode ,发现是vdda 未连接

  7. 单片机学习(八)点阵LED的使用

    目录 一.点阵LED的理论知识 1. 点阵屏的打开方式 2. LED点阵简介 3. 点阵LED的结构和操作方式 4. 74HC595模块 PPT上的简介 串行转并行的工作原理 二.编码实现 1. C5 ...

  8. NOIP 模拟 $22\; \rm e$

    题解 对于这个 \(abs\) 就是求大于 \(r\) 的最小值,小于 \(r\) 的最大值,建权值线段树或平衡树. 因为是 \(k\) 个点的联通块,就是求它们的 \(lca\) 到它们的链,可持久 ...

  9. mpu6050控制舵机云台

    准备材料:2个舵机mg90,云台支架,1个arduino-uno,mpu6050 编程工具:VScode-platformio  or  ArduinoIDE 我使用的是VScode在编辑代码上会更方 ...

  10. 数据结构与算法-排序(十)桶排序(Bucket Sort)

    摘要 桶排序和基数排序类似,相当于基数排序的另外一种逻辑.它是将取值范围当做创建桶的数量,桶的长度就是序列的大小.通过处理比较元素的数值,把元素放在桶的特定位置,然后遍历桶,就可以得到有序的序列. 逻 ...