在C#中动态访问对象属性时,用表达式树可以获得高效性能

在C#中如何用表达式树动态访问对象属性的问题。用户可能已经知道反射的基本用法,但想用表达式树来提高性能,因为表达式树编译后的委托执行速度比反射快。
首先,表达式树的基本概念。表达式树允许在运行时构建代码,并编译成可执行的委托,这样避免了反射带来的性能开销。用户需要一个例子,所以我要构造一个简单的类,比如Person,然后动态获取它的属性,比如Name。
接下来,需要考虑如何构建表达式树。访问属性通常涉及参数表达式、属性访问表达式,以及可能的转换(如果属性是值类型需要装箱)。步骤大致是:创建参数表达式,生成属性访问表达式,然后转换成Lambda表达式,最后编译成委托。
然后,需要处理泛型方法,因为用户可能希望这个方法适用于任何对象和属性类型。所以方法应该是泛型的,比如GetPropertyAccessor<T, TProp>,这样类型安全,同时减少装箱拆箱。
还要考虑错误处理,比如属性是否存在,但用户可能暂时不需要,例子中可以忽略,但实际使用时可能需要添加。
然后,编写示例代码,包括Person类,创建表达式树,编译成委托,并演示如何使用。用户可能还需要比较性能,所以可以提到与反射的性能对比,但例子中可以不实现。
最后,检查代码是否正确,比如Expression.Property的正确使用,Lambda表达式的参数是否正确,以及编译后的委托是否正确调用。
可能遇到的坑:值类型属性需要转换为object,所以可能需要Expression.Convert。例如,如果属性是int类型,直接返回object需要转换,否则会出错。但如果是泛型方法返回TProp,可能不需要,但如果是返回object,则需要处理。
在用户例子中,返回的是object,所以可能需要Convert表达式。但用户提供的示例代码可能没有处理,需要检查是否正确。比如,如果TProp是值类型,访问时需要转换为object,否则委托返回的是值类型,导致装箱问题。因此,应该在表达式树中添加Convert。
不过,在用户提供的示例代码中,GetPropertyAccessor返回Func<object, object>,所以当TProp是值类型时,Expression.Property会返回值类型,需要转换为object。因此,在表达式树中应该用Expression.Convert将属性表达式转换为object类型。
例如:
Expression.Property(param, propertyInfo) 是TProp类型,如果TProp是值类型,直接赋值给object会装箱,所以需要:
Expression.Convert(Expression.Property(param, propertyInfo), typeof(object))
这样,生成的lambda表达式就是Expression<Func<object, object>>,正确处理值类型和引用类型。
但用户提供的示例代码中可能没有这一步,需要检查。比如,原代码中的表达式是:
Expression.Lambda<Func<object, object>>(propertyAccess, param);
而propertyAccess是Expression.Property(param, propertyInfo),此时如果propertyInfo是值类型,则propertyAccess的类型是值类型,无法直接转换为object,所以需要Convert。
因此,正确的代码应该在获取属性后,添加一个Convert表达式。所以,修改后的表达式应该是:
var propertyAccess = Expression.Property(param, propertyInfo);
if (propertyAccess.Type.IsValueType)
{
propertyAccess = Expression.Convert(propertyAccess, typeof(object));
}
或者更简单的方式,不管是否是值类型,都转换为object:
propertyAccess = Expression.Convert(propertyAccess, typeof(object));
但这样可能对于引用类型会多一次转换,不过不影响结果。因此,在构建表达式树时,应该添加Convert。
所以,用户提供的示例代码可能需要调整,添加Convert步骤,否则当属性是值类型时会出错。
总结,正确的表达式树构建步骤应该是:

  1. 创建参数表达式:ParameterExpression param = Expression.Parameter(typeof(object), "x");
  2. 转换参数为具体类型:UnaryExpression convertedParam = Expression.Convert(param, typeof(T));
  3. 访问属性:MemberExpression propertyAccess = Expression.Property(convertedParam, propertyInfo);
  4. 将属性值转换为object(如果需要):UnaryExpression convertedProperty = Expression.Convert(propertyAccess, typeof(object));
  5. 构建Lambda表达式:var lambda = Expression.Lambda<Func<object, object>>(convertedProperty, param);
    但用户希望的是通用的方法,所以可能需要泛型方法,或者正确处理类型转换。
    可能用户提供的例子中没有处理这一点,导致值类型属性无法正确返回。因此,在编写示例时,需要注意添加Convert。
    现在,我需要将这些思考整理成一个清晰的示例,确保代码正确,并解释每一步的作用,同时指出可能的注意事项,比如性能优势和类型转换的问题。
using System;
using System.Linq.Expressions;
using System.Reflection;

public class Person
{
    public string Name { get; set; }
}

public static class PropertyAccessor
{
    // 创建泛型委托缓存(提升性能)
    private static readonly Dictionary<string, Delegate> _accessorCache = new Dictionary<string, Delegate>();

    public static Func<object, object> GetPropertyAccessor(Type objectType, string propertyName)
    {
        var key = $"{objectType.FullName}.{propertyName}";
        if (!_accessorCache.TryGetValue(key, out var accessor))
        {
            // 获取属性信息
            var propertyInfo = objectType.GetProperty(propertyName);
            if (propertyInfo == null) throw new ArgumentException("Property not found");

            // 构建表达式树
            var param = Expression.Parameter(typeof(object), "obj");
            var convertedParam = Expression.Convert(param, objectType); // 转换输入对象为具体类型
            var propertyAccess = Expression.Property(convertedParam, propertyInfo);
            var convertResult = Expression.Convert(propertyAccess, typeof(object)); // 处理值类型装箱
            var lambda = Expression.Lambda<Func<object, object>>(convertResult, param);

            accessor = lambda.Compile();
            _accessorCache[key] = accessor;
        }
        return (Func<object, object>)accessor;
    }
}

// 使用示例
var person = new Person { Name = "Alice" };
var accessor = PropertyAccessor.GetPropertyAccessor(typeof(Person), "Name");
var name = accessor(person); // 返回 "Alice"

注意事项

  1. 线程安全:缓存使用需考虑线程同步
  2. 内存管理:长期运行的应用程序需注意缓存大小
  3. 类型兼容:确保输入对象可转换为目标类型

如果需要支持更复杂的场景(如嵌套属性访问、索引器等),可以通过组合多个表达式节点来实现更复杂的访问逻辑。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/973336.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【单臂路由配置】

【单臂路由配置】 设备接口IP子网网关vlanR1G0/0/1.1192.168.1.254255.255.255.0NAvlan10R1G0/0/1.2192.168.2.254255.255.255.0NAvlan20R1G0/0/1.3192.168.3.254255.255.255.0NAvlan30PC1e0/0/1192.168.1.1255.255.255.0192.168.1.254vlan10PC2e0/0/1192.168.2.1255.255.255.0…

NutUI内网离线部署

文章目录 官网拉取源代码到本地仓库修改源代码打包构建nginx反向代理部署访问内网离线地址 在网上找了一圈没有写NutUI内网离线部署的文档&#xff0c;花了1天时间研究下&#xff0c;终于解决了。 对于有在内网离线使用的小伙伴就可以参考使用了 如果还是不会联系UP主:QQ:10927…

【Linux AnolisOS】关于Docker的一系列问题。尤其是拉取东西时的网络问题,镜像源问题。

AnolisOS 8中使用Docker部署&#xff08;全&#xff09;_anolis安装docker-CSDN博客 从在虚拟机安装龙蜥到安装docker上面这篇文章写的很清晰了&#xff0c;我重点讲述我解决文章里面问题一些的方法。 问题1&#xff1a; docker: Get https://registry-1.docker.io/v2/: net/h…

免费体验,在阿里云平台零门槛调用满血版DeepSeek-R1模型

一、引言 随着人工智能技术的飞速发展&#xff0c;各类AI模型层出不穷。其中&#xff0c;DeepSeek作为一款新兴的推理模型&#xff0c;凭借其强大的技术实力和广泛的应用场景&#xff0c;逐渐在市场中崭露头角。本文将基于阿里云提供的零门槛解决方案&#xff0c;对DeepSeek模…

ARM Linux平台下 OpenCV Camera 实验

一、硬件原理 1. OV2640 1.1 基本功能 OV2640 是一款低功耗、高性能的图像传感器&#xff0c;支持以下功能&#xff1a; 最高分辨率&#xff1a;200 万像素&#xff08;1600x1200&#xff09;。 输出格式&#xff1a;JPEG、YUV、RGB。 内置图像处理功能&#xff1a;自动曝…

论文笔记-WSDM2025-ColdLLM

论文笔记-WSDM2025-Large Language Model Simulator for Cold-Start Recommendation ColdLLM&#xff1a;用于冷启动推荐的大语言模型模拟器摘要1.引言2.前言3.方法3.1整体框架3.1.1行为模拟3.1.2嵌入优化 3.2耦合漏斗ColdLLM3.2.1过滤模拟3.2.2精炼模拟 3.3模拟器训练3.3.1LLM…

后端开发:开启技术世界的新大门

在互联网的广阔天地中&#xff0c;后端开发宛如一座大厦的基石&#xff0c;虽不直接与用户 “面对面” 交流&#xff0c;却默默地支撑着整个互联网产品的稳定运行。它是服务器端编程的核心领域&#xff0c;负责处理数据、执行业务逻辑以及与数据库和其他后端服务进行交互。在当…

如何查看java的字节码文件?javap?能用IDEA吗?

编译指令&#xff1a; javac YourProject.java 查看字节码文件的指令&#xff1a; javap -c -l YourProject.class 不添加-c指令就不会显示字节码文件&#xff1a; 不添加 -l 就不会显示源代码和字节码文件的对应关系&#xff1a; 添加-l之后多出来这些&#xff1a; IDEA不太…

Linux-GlusterFS配置

文章目录 GlusterFS配置 &#x1f3e1;作者主页&#xff1a;点击&#xff01; &#x1f916;Linux专栏&#xff1a;点击&#xff01; ⏰️创作时间&#xff1a;2025年02月18日19点21分 GlusterFS配置 1、分区操作 fdisk -l #查看本地磁盘 fdisk /dev/vdb #对/dev/vdb进…

[C语言]指针进阶压轴题

下面代码打印结果是什么&#xff1f; #include<stdio.h> int main() {char* c[] { "ENTER","NEW","POINT","FIRST" };char** cp[] { c 3,c 2,c 1,c };char*** cpp cp;printf("%s\n", **cpp);printf("%s\n…

DeepSeek 助力 Vue 开发:打造丝滑的点击动画(Click Animations)

前言&#xff1a;哈喽&#xff0c;大家好&#xff0c;今天给大家分享一篇文章&#xff01;并提供具体代码帮助大家深入理解&#xff0c;彻底掌握&#xff01;创作不易&#xff0c;如果能帮助到大家或者给大家一些灵感和启发&#xff0c;欢迎收藏关注哦 &#x1f495; 目录 Deep…

SpringBoot论坛网站 – 功能详解与部署教程

项目概述 《SpringBoot的论坛网站》是一个基于SpringBoot框架开发的现代化论坛平台&#xff0c;旨在为用户提供一个便捷的交流空间。该项目不仅功能丰富&#xff0c;还具备良好的扩展性和易用性&#xff0c;适合用于学习、分享和讨论各类话题。以下是项目的核心功能模块和部署…

SpringSecurity初始化的本质

一、对SpringSecurity初始化的几个疑问 通过前面第一次请求访问的分析我们明白了一个请求就来后的具体处理流程 对于一个请求到来后会通过FilterChainProxy来匹配一个对应的过滤器链来处理该请求。那么这里我们就有几个疑惑。 FilterChainProxy什么时候创建的?过滤器链和对应的…

【大模型系列篇】DeepSeek-R1如何通过强化学习有效提升大型语言模型的推理能力?

如何通过强化学习&#xff08;RL&#xff09;有效提升大型语言模型&#xff08;LLM&#xff09;的推理能力&#xff1f; 《DeepSeek-R1: Incentivizing Reasoning Capability in LLMs via Reinforcement Learning》由DeepSeek-AI团队撰写&#xff0c;主要介绍了他们开发的第一代…

无缝对接[系列2]:在VSCode中继续接入本地DeepSeek的完整指南---实现代码协助编写~

无缝对接&#xff1a;在VSCode中继续接入本地DeepSeek的完整指南 在上一篇文章中&#xff0c;我们已经详细介绍了如何成功部署本地 DeepSeek&#xff0c;使得开发者能够在本地环境中高效地进行深度学习和数据分析工作。 部署完成后&#xff0c;如何在 Visual Studio Code (VS…

linux5-多任务--进程fork()

一.多任务&#xff1a;让系统具备同时处理多个任务的能力 1.如何实现多任务 1.1进程&#xff1a;操作系统上正在运行的程序&#xff0c;需要消耗内存和CPU 1.1.1 进程的生存周期&#xff1a;创建、调度、消亡 1.1.1.1进程的创建&#xff1a;每个进程被创建时&#xff0c;操作…

原生稀疏注意力机制(NSA):硬件对齐且可原生训练的稀疏注意力机制-论文阅读

摘要 长上下文建模对于下一代语言模型至关重要&#xff0c;但标准注意力机制的高计算成本带来了巨大的计算挑战。稀疏注意力提供了一种在保持模型能力的同时提高效率的有前途的方向。本文提出了一种名为 NSA&#xff08;原生可训练稀疏注意力机制&#xff09; 的方法&#xff…

C++ Primer 库-IO类

欢迎阅读我的 【CPrimer】专栏 专栏简介&#xff1a;本专栏主要面向C初学者&#xff0c;解释C的一些基本概念和基础语言特性&#xff0c;涉及C标准库的用法&#xff0c;面向对象特性&#xff0c;泛型特性高级用法。通过使用标准库中定义的抽象设施&#xff0c;使你更加适应高级…

FlutterAssetsGenerator插件的使用

在Plugins中找到FlutterAssetsGenerator插件&#xff0c;点击安装。 更改生成的资源索引类可以修改名字。 在根目录下创建assets/images文件夹&#xff0c;用于存储图片。 点击images文件夹&#xff0c;鼠标右键点击Flutter&#xff1a;Configuring Paths&#xff0c;pub…

网络安全与防范

1.重要性 随着互联网的发达&#xff0c;各种WEB应用也变得越来越复杂&#xff0c;满足了用户的各种需求&#xff0c;但是随之而来的就是各种网络安全的问题。了解常见的前端形式和保护我们的网站不受干扰是我们每个优秀fronter必备的技能。 2.分类 XSS干扰 CSRF干扰 网络劫持干…