首页 > 编程语言 >C#高性能动态获取对象属性值的步骤

C#高性能动态获取对象属性值的步骤

时间:2023-04-27 22:11:45浏览次数:39  
标签:MemberGetDelegate Name People C# 步骤 people 高性能 stopwatch type

动态获取对象的性能值,这个在开发过程中经常会遇到,这里我们探讨一下何如高性能的获取属性值。为了对比测试,我们定义一个类People

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

然后通过直接代码调用方式来取1千万次看要花多少时间:

private static void Directly()
{
  People people = new People { Name = "Wayne" };
  Stopwatch stopwatch = Stopwatch.StartNew();
  for (int i = 0; i < 10000000; i++)
  {
    object value = people.Name;
  }
  stopwatch.Stop();
  Console.WriteLine("Directly: {0}ms", stopwatch.ElapsedMilliseconds);
}

大概花了37ms:

反射
通过反射来获取对象的属性值,这应该是大家常用的方式,但这种方式的性能比较差。接下来我们来看看同样取1千万次需要多少时间:

private static void Reflection()
{
  People people = new People { Name = "Wayne" };
  Type type = typeof(People);
  PropertyInfo property = type.GetProperty("Name");
  Stopwatch stopwatch = Stopwatch.StartNew();
  for (int i = 0; i < 10000000; i++)
  {
    object value = property.GetValue(people);
  }
  stopwatch.Stop();
  Console.WriteLine("Reflection: {0}ms", stopwatch.ElapsedMilliseconds);
}

大概花了1533ms,果然要慢很多:

那既然反射慢,那还有没有其它方式呢?

动态构建Lambda
我们知道可以动态构建Linq的Lambda表达式,然后通过编译后得到一个委托,如果能动态构建返回属性值的委托,就可以取到值了。所以我们想办法构建一个像这样的委托:

Func<People, object> getName = m => m.Name;

接下来我们就通过Expression来构建:

private static void Lambda()
{
  People people = new People { Name = "Wayne" };
  Type type = typeof(People);
  var parameter = Expression.Parameter(type, "m");//参数m
  PropertyInfo property = type.GetProperty("Name");
  Expression expProperty = Expression.Property(parameter, property.Name);//取参数的属性m.Name
  var propertyDelegateExpression = Expression.Lambda(expProperty, parameter);//变成表达式 m => m.Name
  var propertyDelegate = (Func<People, object>)propertyDelegateExpression.Compile();//编译成委托
  Stopwatch stopwatch = Stopwatch.StartNew();
  for (int i = 0; i < 10000000; i++)
  {
    object value = propertyDelegate.Invoke(people);
  }
  stopwatch.Stop();
  Console.WriteLine("Lambda:{0}ms", stopwatch.ElapsedMilliseconds);
}

然后我们测试一下,大概花了138ms,性能要比反射好非常多:

委托调用

虽然动态构建Lambda的性能已经很好了,但还是更好吗?毕竟比直接调用还是差了一些,要是能直接调用属性的取值方法就好了。

在C#中,可读属性都有一个对应的get_XXX()的方法,可以通过调用这个方法来取得对应属性的值。可以使用System.Delegate.CreateDelegate创建一个委托来调用这个方法。

通过委托调用方法来取得属性值
我们定义一个MemberGetDelegate的委托,然后通过它来调用取值方法:

delegate object MemberGetDelegate(People p);
private static void Delegate()
{
  People people = new People { Name = "Wayne" };
  Type type = typeof(People);
  PropertyInfo property = type.GetProperty("Name");
  MemberGetDelegate memberGet = (MemberGetDelegate)System.Delegate.CreateDelegate(typeof(MemberGetDelegate), property.GetGetMethod());
  Stopwatch stopwatch = Stopwatch.StartNew();
  for (int i = 0; i < 10000000; i++)
  {
    object value = memberGet(people);
  }
  stopwatch.Stop();
  Console.WriteLine("Delegate: {0}ms", stopwatch.ElapsedMilliseconds);
}

然后我们测试一下,大概花了38ms,性能几乎与直接调用一致:

最后做一个简单的封装,缓存一下创建的Delegate

public class PropertyValue<T>
{
  private static ConcurrentDictionary<string, MemberGetDelegate> _memberGetDelegate = new ConcurrentDictionary<string, MemberGetDelegate>();
  delegate object MemberGetDelegate(T obj);
  public PropertyValue(T obj)
  {
    Target = obj;
  }
  public T Target { get; private set; }
  public object Get(string name)
  {
    MemberGetDelegate memberGet = _memberGetDelegate.GetOrAdd(name, BuildDelegate);
    return memberGet(Target);
  }
  private MemberGetDelegate BuildDelegate(string name)
  {
    Type type = typeof(T);
    PropertyInfo property = type.GetProperty(name);
    return (MemberGetDelegate)Delegate.CreateDelegate(typeof(MemberGetDelegate), property.GetGetMethod());
  }
}
public class PropertyValue<T>
{
  private static ConcurrentDictionary<string, MemberGetDelegate> _memberGetDelegate = new ConcurrentDictionary<string, MemberGetDelegate>();
  delegate object MemberGetDelegate(T obj);
  public PropertyValue(T obj)
  {
    Target = obj;
  }
  public T Target { get; private set; }
  public object Get(string name)
  {
    MemberGetDelegate memberGet = _memberGetDelegate.GetOrAdd(name, BuildDelegate);
    return memberGet(Target);
  }
  private MemberGetDelegate BuildDelegate(string name)
  {
    Type type = typeof(T);
    PropertyInfo property = type.GetProperty(name);
    return (MemberGetDelegate)Delegate.CreateDelegate(typeof(MemberGetDelegate), property.GetGetMethod());
  }
}

这样使用起来就方便多了

People people = new People { Name = "Wayne" };
PropertyValue<People> propertyValue = new PropertyValue<People>(people);
object value = propertyValue.Get("Name");

 

标签:MemberGetDelegate,Name,People,C#,步骤,people,高性能,stopwatch,type
From: https://www.cnblogs.com/wl-blog/p/17360382.html

相关文章

  • #yyds干货盘点# LeetCode程序员面试金典:外观数列
    题目:给定一个正整数n,输出外观数列的第n项。「外观数列」是一个整数序列,从数字1开始,序列中的每一项都是对前一项的描述。你可以将其视作是由递归公式定义的数字字符串序列:countAndSay(1)="1"countAndSay(n)是对countAndSay(n-1)的描述,然后转换成另一个数字字符串。前五项......
  • 使用CGLIB生成代理
    知识点【使用前提条件:【/**如果这个代理的类没有实现接口就不能使用JDK中的动态代理*这时需要使用第三方的.jarCGLIB实现代理**/】publicclassCGLIBProxyimplementsMethodInterceptor{privateObjecttar......
  • 通过在classpath自动扫描方式把组件纳入spring容器中管理
    知识点:【前面的例子我们都是使用XML的bean定义来配置组件。在一个稍大的项目中,通常会有上百个组件,如果这些这组件采用xml的bean定义来配置,显然会增加配置文件的体积,查找及维护起来也不太方便。spring2.5为我们引入了组件自动扫描机制,他可以在类路径底......
  • 横竖屏切换时候activity的生命周期
    1、不设置Activity的android:configChanges时,切屏会重新调用各个生命周期,切横屏时会执行一次,切竖屏时会执行两次2、设置Activity的android:configChanges="orientation"时,切屏还是会重新调用各个生命周期,切横、竖屏时只会执行一次3、设置Activity的android:configChanges=......
  • mybatis控制动态SQL拼接标签之foreach标签
    mybatis控制动态SQL拼接标签之foreach标签foreach标签主要用于构建in条件,可在sql中对集合进行迭代。也常用到批量删除、添加等操作中。这个标签在实际业务中非常常用,当然运维旧项目也会发现,有些坑,用java循环执行sql来表示批量插入。属性说明:collection:collection属性的值有三......
  • 4/27打卡 stl初识vector
    1#include<iostream>2#include<vector>3usingnamespacestd;456voidtest()7{8vector<vector<int>>v;9vector<int>v1;10vector<int>v2;11vector<int>v3;12vector<in......
  • Async/Await替代Promise的6个理由
     Node.js的异步编程方式有效提高了应用性能;然而回调地狱却让人望而生畏,Promise让我们告别回调函数,写出更优雅的异步代码;在实践过程中,却发现Promise并不完美;技术进步是无止境的,这时,我们有了Async/Await。 [Async/Await替代Promise的6个理由|Fundebug博客-](https://blo......
  • 文件上传下载-SpringMvc
    进行文件上传时,表单需要做的准备:1.请求方式为POST:<formaction=”uploadServlet”method=”post”/>2.使用file的表单域:<inputtype=”file”name=”file”/>3.使用multipart/form-data的请求编码方式:<formaction=”uploadServlet”type=”file”name=”file”metho......
  • 2022-04-27:用go语言重写ffmpeg的remuxing.c示例。
    2022-04-27:用go语言重写ffmpeg的remuxing.c示例。答案2022-04-27:ffmpeg的remuxing.c是一个用于将多媒体文件从一种容器格式转换为另一种容器格式的命令行工具。它可以将音频、视频和字幕等元素从源文件中提取出来,并按照用户指定的方式重新封装到目标文件中。在本篇文章中,我将对ffmp......
  • [犯病记] 重新安装libc.so.6
    [犯病记]重新安装libc.so.6背景:有一天,我犯了个病,试图在一个机器上安装DOCA,然后安装程序就下了一堆库,把机器上的环境搞得一团糟,原来的DPDK也不能用了。DOCA的安装程序不仅重新安装了一大堆库,还更换了系统内核版本,修改了网卡配置,以及更换了网卡驱动(MLNX_OFED)。为此我不得不将上述......