C# 中的执行表达式树(Expression Tree)
引言:
在C#编程中,表达式树(Expression Tree)是一种强大的工具,用于表示和执行计算表达式。表达式树将计算表达式抽象为树状结构,每个节点代表表达式中的一个元素,如常量、变量、方法调用等。本文将深入探讨表达式树的基本概念、创建方法、修改和删除节点、查询和遍历技巧以及在C#中的应用示例。通过学习这些内容,您将能够更好地理解和利用表达式树来提升您的编程技能。
1.表达式树的基本概念及其在C#中的定义和使用
表达式树是一种树形数据结构,它代表了代码中的计算表达式。在C#中,表达式树用于捕获复杂的计算逻辑,并可以用于诸如LINQ查询、数据绑定、反射等场景。表达式树将计算表达式抽象成树状结构,每个节点代表表达式中的一个元素,如常量、变量、方法调用等。这种结构使得表达式易于分析和转换,同时也为动态生成代码和进行运行时分析提供了便利。
2.创建表达式树的方法,包括使用运算符和表达式生成器
在C#中,我们可以使用System.Linq.Expressions命名空间下的Expression类来创建表达式树。通过构建表达式树,我们可以灵活地构建各种计算表达式,而不必编写冗长的代码。例如,要创建一个加法表达式,我们可以使用以下代码:
Expression<Func<int, int, int>> add = (x, y) => x + y;
在上面的代码中,我们定义了一个名为add的lambda表达式,它接受两个int类型的参数并返回它们的和。这个表达式就是一个加法表达式树,其中包含两个变量节点和一个乘法节点。
2.1使用运算符创建表达式树
// 创建参数表达式
ParameterExpression left = Expression.Parameter(typeof(int), "left");
ParameterExpression right = Expression.Parameter(typeof(int), "right");// 创建加法表达式
BinaryExpression sum = Expression.Add(left, right);
2.2 使用表达式生成器创建表达式树
csharp
// 创建参数表达式
ParameterExpression input = Expression.Parameter(typeof(int), "input");// 创建平方表达式
UnaryExpression square = Expression.Power(input, Expression.Constant(2));
除了使用lambda表达式,我们还可以使用表达式生成器来创建表达式树。表达式生成器是一种API,它允许我们动态地构建表达式树。以下是一个使用表达式生成器创建乘法表达式树的示例:
var left = Expression.Constant(3);
var right = Expression.Constant(4);
var multiplication = Expression.Multiply(left, right);
在上面的代码中,我们使用表达式生成器创建了两个常量节点,并将它们相乘。
3.修改和删除表达式树中的节点
表达式树是可变的,我们可以修改和删除树中的节点。以下是一个修改表达式树中节点的示例:
var originalExpression = Expression.Multiply(left, right);
var modifiedExpression = originalExpression.Update(left, Expression.Constant(5));
在上面的代码中,我们修改了乘法表达式树中的左节点,将其值更改为5。
要删除表达式树中的节点,我们可以使用以下代码:
var parent = originalExpression.RemoveNode(originalExpression.Body);
在上面的代码中,我们删除了原始表达式树中的根节点。
4.查询和遍历表达式树
使用递归下降和深度优先搜索 查询和遍历表达式树是表达式树操作的常见任务。以下是一个使用递归下降方法遍历表达式树的示例:
void Visit(Expression node)
{if (node is BinaryExpression binary){Visit(binary.Left);Visit(binary.Right);}else if (node is UnaryExpression unary){Visit(unary.Operand);}else if (node is ConstantExpression constant){Console.WriteLine($"常量值:{constant.Value}");}// ... 其他节点类型的处理逻辑
}
在上面的代码中,我们定义了一个Visit方法,它接受一个表达式节点作为参数,并递归地遍历表达式树。
除了递归下降方法,我们还可以使用深度优先搜索(DFS)算法来遍历表达式树。以下是一个使用DFS遍历表达式树的示例:
void DfsTraversal(Expression node)
{if (node == null)return;Console.WriteLine($"节点类型:{node.NodeType}");if (node is BinaryExpression binary){DfsTraversal(binary.Left);DfsTraversal(binary.Right);}else if (node is UnaryExpression unary){DfsTraversal(unary.Operand);}// ... 其他节点类型的处理逻辑}
在上面的代码中,我们定义了一个DfsTraversal方法,它接受一个表达式节点作为参数,并使用深度优先搜索算法遍历表达式树。
5. 表达式树在C#中的实际应用例子
解析XML和JSON数据.表达式树在C#中有着广泛的应用,以下是一些具体的示例:
5.1解析XML数据
XML是一种常用的数据交换格式,表达式树可以用于解析XML数据。以下是一个使用表达式树解析XML数据的示例:
var xml = @"<root><person id='1'><name>张三</name><age>30</age></person><person id='2'><name>李四</name><age>40</age></person></root>";
var xmlDocument = new XmlDocument();
xmlDocument.LoadXml(xml);
var expression = Expression.ParseLambda<Func<XmlNode, bool>>(@"param => param.Name == 'name' && param.Attributes['id'].Value == '1'",typeof(XmlNode));
var query = xmlDocument.XPathSelectNodes(expression);
在上面的代码中,我们使用表达式树构建了一个查询条件,它选择id为1的person元素的name属性。然后,我们使用XPathSelectNodes方法根据表达式树查询XML数据。
5.2解析JSON数据
JSON是另一种常用的数据交换格式,表达式树同样可以用于解析JSON数据。以下是一个使用表达式树解析JSON数据的示例:
var json = @"{'name': '张三', 'age': 30, 'city': '北京'}";
var jsonDocument = JsonDocument.Parse(json);
var expression = Expression.ParseLambda<Func<JsonElement, bool>>(@"param => param.Value<string>(""name"") == ""张三"" && param.Value<int>(""age"") == 30",typeof(JsonElement));
var query = jsonDocument.RootElement.EnumerateArray().Where(expression).ToArray();
在上面的代码中,我们使用表达式树构建了一个查询条件,它选择name为张三且age为30的JSON元素。然后,我们使用EnumerateArray方法和Where方法根据表达式树查询JSON数据。
结论
表达式树是C#编程中的一个强大特性,它提供了一种灵活、高效的方式来表示和执行计算表达式。通过理解表达式树的基本概念、创建方法、修改和删除节点、查询和遍历技巧以及在C#中的应用示例,您可以更好地理解和利用表达式树来提升您的编程技能。无论是在解析XML和JSON数据、LINQ查询、数据绑定、反射还是其他领域,表达式树都是一个非常有用的工具。
相关文章:
C# 中的执行表达式树(Expression Tree)
引言: 在C#编程中,表达式树(Expression Tree)是一种强大的工具,用于表示和执行计算表达式。表达式树将计算表达式抽象为树状结构,每个节点代表表达式中的一个元素,如常量、变量、方法调用等。本…...
森林监测VR虚拟情景再现系统更便利
AI人工智能技术已经逐渐渗透到各个领域,为我们的生活带来了诸多便利。在虚拟仿真教学领域,AI技术的应用也日益丰富,为虚拟情景交互体验带来了前所未有的好处。 提高VR虚拟情景的逼真度 通过深度学习和计算机视觉等技术,AI/VR虚拟现…...
高频面试题整理(一)
文章目录 平台无关性如何实现?JVM如何加载 .class文件?什么是反射?谈谈ClassLoader谈谈类的双亲委派机制类的加载方式Java的内存模型?JVM内存模型-jdk8程序计数器:Java虚拟机栈局部变量表和操作数栈: Java内存模型中堆和栈的区别…...
2-23 switch、JVM内存模型、垃圾回收机制、this、static、变量的分类
文章目录 switch 实现成绩评级JVM内存模型概念栈的特点堆的特点 垃圾回收机制通用的分代垃圾回收机制三种清理算法垃圾回收过程垃圾回收常见的两种检测引用算法内存泄露常见原因 this的用法创建对象的四步 static 静态特点 变量的分类和作用域import switch 实现成绩评级 switc…...
基础!!!吴恩达deeplearning.ai:卷积层
以下内容有任何不理解可以翻看我之前的博客哦:吴恩达deeplearning.ai专栏 文章目录 回顾——密集层 Dense Layer卷积层 Convolutional Neural Network定义优势具体说明心电图卷积层搭建 到目前为止,你使用的所有神经网络层都是密集层类型,这…...
SpringBoot案例(黑马学习笔记)
这个案例呢,就是Tlias智能学习辅助系统。 参考接口文档完成后端功能的开 发,然后结合前端工程进行联调测试即可。 完成后的成品效果展示: 准备工作 需求&环境搭建 需求说明 部门管理 部门管理功能开发包括: ● 查询部门列…...
项目流程图
实现便利店自助付款项目 服务器: 1、并发服务器(多进程、多线程、IO多路复用) 2、SQL数据库的创建和使用(增删改查) 3、以模块化编写项目代码,按照不同模块编写.h/.c文件 客户端: 1、QT客户端界…...
鸿蒙这么大声势,为何迟迟看不见岗位?最新数据来了
对于鸿蒙生态建设而言,2024年可谓至关重要,而生态建设的前提,就是要有足够的开发人才。与之对应的,今年春招市场上与鸿蒙相关岗位和人才旺盛的热度,一方面反应了鸿蒙生态的逐渐壮大,另一方面也让人们对鸿蒙…...
Qt中关于信号与槽函数的思考
信号与槽函数的思考 以pushbutton控件为例,在主界面上放置一个pushbutton控件,点击右键选择关联槽函数,关联一个click函数,如下图所示: 在该函数中,实现了一个点击pushbutton按钮后,弹出一个窗…...
项目技术栈-解决方案-消息队列
项目技术栈-解决方案-消息队列 概念应用场景1. 异步处理 参考文章消息队列(Message Queue) 概念 “消息”是在两台计算机间传送的数据单位。 消息可以非常简单,例如只包含文本字符串; 也可以更复杂 ,包括对象等。 队…...
【深度优先搜索】【图论】【推荐】332. 重新安排行程
作者推荐 动态规划的时间复杂度优化 本文涉及知识点 深度优先搜索 图论 LeetCode332. 重新安排行程 给你一份航线列表 tickets ,其中 tickets[i] [fromi, toi] 表示飞机出发和降落的机场地点。请你对该行程进行重新规划排序。 所有这些机票都属于一个从 JFK&a…...
DAY9-防病毒AV概述
DNS过滤 URL过滤和DNS过滤对比...
TCP缓存
TCP缓存是指TCP协议在数据传输过程中使用的一种机制,用于临时存储和管理数据包。它主要有三个作用:提高网络性能、保证数据的可靠性和实现流量控制。 首先,TCP缓存可以提高网络性能。当发送端发送数据时,TCP协议会将数据分割成若…...
Socket网络编程(一)——网络通信入门基本概念
目录 网络通信基本概念什么是网络?网络通信的基本架构什么是网络编程?7层网络模型-OSI模型什么是Socket?Socket的作用和组成Socket传输原理Socket与TCP、UDP的关系CS模型(Client-Server Application)报文段牛刀小试(TCP消息发送与接收&#…...
RTCA DO-178C 机载系统和设备认证中的软件注意事项-软件质量保证流程(八)
8.0 软件质量保证流程 SOFTWARE QUALITY ASSURANCE PROCESS 本节讨论软件质量保证 (SQA) 过程的目标和活动。 SQA 流程按照软件规划流程(参见 4)和软件质量保证计划(参见 11.5)的定义进行应用。 SQA 过程活动的输出记录在软件质量…...
K 个一组翻转链表 力扣
【玩转校招算法面试】第三天:链表中的节点每k个一组翻转(动画演示、手写 Java 代码、详细注释、LeetCode 高频算法题)_哔哩哔哩_bilibili 初始状态:1 -> 2 -> 3,pre null, cur 1保存当前节点的下一个节点&…...
Java毕业设计 基于SSM SpringBoot vue购物比价网站
Java毕业设计 基于SSM SpringBoot vue购物比价网站 SSM vue 购物比价网站 功能介绍 首页 图片轮播 商品 商品分类 商品详情 评论 收藏 商品攻略 商品信息 公告栏 在线反馈 登录 注册 个人中心 我的收藏 后台管理 登录 注册商品户 个人中心 修改密码 个人信息 商品户管理 用户…...
Linux按键输入实验-按键功能完善
一. 简介 前面一篇文章实现了 按键的字符设备驱动代码框架,文章地址如下: Linux按键输入实验-按键的字符设备驱动代码框架-CSDN博客 本文在 字符设备驱动框架实现的基础上,加入按键GPIO的初始化功能。 二. Linux按键输入实验-按键的GPIO…...
二分查找讲解
关于我为什么要写单独开一篇文章写二分,实际上那么多困难的算法,比如线段树,并查集等等都没有难倒我,我最近却被二分难倒了,而且是两次,两次在赛场上做不出来二分的应用题,于是我决定写一篇二分查找的算法总结.刚接触算法的时候本来是要写一篇的,但后面因为各种原因搁置了,现在…...
跨区域复制建筑UI输入框脚本迷你世界
--复制区域文件 --设置坐标起点,终点 --创建区域 --获取坐标id,data --星空露珠工作室制作 local pos1{x-16,y7,z28} local pos2{x28,y44,z-9} local block{num0} local str{} local str0{} local num0 local count0 local ui6 --几个输入框 local romath.random(…...
基于距离变化能量开销动态调整的WSN低功耗拓扑控制开销算法matlab仿真
目录 1.程序功能描述 2.测试软件版本以及运行结果展示 3.核心程序 4.算法仿真参数 5.算法理论概述 6.参考文献 7.完整程序 1.程序功能描述 通过动态调整节点通信的能量开销,平衡网络负载,延长WSN生命周期。具体通过建立基于距离的能量消耗模型&am…...
HBuilderX安装(uni-app和小程序开发)
下载HBuilderX 访问官方网站:https://www.dcloud.io/hbuilderx.html 根据您的操作系统选择合适版本: Windows版(推荐下载标准版) Windows系统安装步骤 运行安装程序: 双击下载的.exe安装文件 如果出现安全提示&…...
前端开发面试题总结-JavaScript篇(一)
文章目录 JavaScript高频问答一、作用域与闭包1.什么是闭包(Closure)?闭包有什么应用场景和潜在问题?2.解释 JavaScript 的作用域链(Scope Chain) 二、原型与继承3.原型链是什么?如何实现继承&a…...
【Java学习笔记】BigInteger 和 BigDecimal 类
BigInteger 和 BigDecimal 类 二者共有的常见方法 方法功能add加subtract减multiply乘divide除 注意点:传参类型必须是类对象 一、BigInteger 1. 作用:适合保存比较大的整型数 2. 使用说明 创建BigInteger对象 传入字符串 3. 代码示例 import j…...
《C++ 模板》
目录 函数模板 类模板 非类型模板参数 模板特化 函数模板特化 类模板的特化 模板,就像一个模具,里面可以将不同类型的材料做成一个形状,其分为函数模板和类模板。 函数模板 函数模板可以简化函数重载的代码。格式:templa…...
C++.OpenGL (14/64)多光源(Multiple Lights)
多光源(Multiple Lights) 多光源渲染技术概览 #mermaid-svg-3L5e5gGn76TNh7Lq {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-3L5e5gGn76TNh7Lq .error-icon{fill:#552222;}#mermaid-svg-3L5e5gGn76TNh7Lq .erro…...
Mysql中select查询语句的执行过程
目录 1、介绍 1.1、组件介绍 1.2、Sql执行顺序 2、执行流程 2.1. 连接与认证 2.2. 查询缓存 2.3. 语法解析(Parser) 2.4、执行sql 1. 预处理(Preprocessor) 2. 查询优化器(Optimizer) 3. 执行器…...
iOS性能调优实战:借助克魔(KeyMob)与常用工具深度洞察App瓶颈
在日常iOS开发过程中,性能问题往往是最令人头疼的一类Bug。尤其是在App上线前的压测阶段或是处理用户反馈的高发期,开发者往往需要面对卡顿、崩溃、能耗异常、日志混乱等一系列问题。这些问题表面上看似偶发,但背后往往隐藏着系统资源调度不当…...
Java数值运算常见陷阱与规避方法
整数除法中的舍入问题 问题现象 当开发者预期进行浮点除法却误用整数除法时,会出现小数部分被截断的情况。典型错误模式如下: void process(int value) {double half = value / 2; // 整数除法导致截断// 使用half变量 }此时...
【JavaSE】多线程基础学习笔记
多线程基础 -线程相关概念 程序(Program) 是为完成特定任务、用某种语言编写的一组指令的集合简单的说:就是我们写的代码 进程 进程是指运行中的程序,比如我们使用QQ,就启动了一个进程,操作系统就会为该进程分配内存…...
