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

 更新时间:2020年12月09日 08:38:53   作者:ZKEASOFT  
这篇文章主要介绍了C#高性能动态获取对象属性值的步骤,帮助大家更好的理解和使用c#,感兴趣的朋友可以了解下

动态获取对象的性能值,这个在开发过程中经常会遇到,这里我们探讨一下何如高性能的获取属性值。为了对比测试,我们定义一个类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());
  }
}

这样使用起来就方便多了

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

以上就是C#高性能动态获取对象属性值的步骤的详细内容,更多关于c# 获取对象属性值的资料请关注脚本之家其它相关文章!

相关文章

  • C# 中的EventHandler实例详解

    C# 中的EventHandler实例详解

    本文通过案例实例介绍了c#中的eventhandler,需要的的朋友参考下吧
    2017-04-04
  • 利用FlubuCore用C#来写DevOps脚本的方法详解

    利用FlubuCore用C#来写DevOps脚本的方法详解

    这篇文章主要介绍了利用FlubuCore用C#来写DevOps脚本的方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-07-07
  • C#连接Mysql实现增删改查的操作

    C#连接Mysql实现增删改查的操作

    在IT行业中,数据库连接是应用程序开发中的重要环节,尤其是在使用C#进行Windows或者Web应用开发时,经常需要与各种数据库进行交互,其中就包括广泛使用的MySQL,本篇将详细讲解如何使用C#语言来连接MySQL数据库,以实现数据的读取、写入和其他操作
    2024-09-09
  • C#静态变量与实例变量实例分析

    C#静态变量与实例变量实例分析

    这篇文章主要介绍了C#静态变量与实例变量,是深入掌握PHP程序设计所必须掌握的重要概念,需要的朋友可以参考下
    2014-09-09
  • C#中限制并发任务数量的高效方法与技巧分享

    C#中限制并发任务数量的高效方法与技巧分享

    在C#中,处理并发操作是一项常见且强大的功能,尤其是在需要执行多个任务但又希望限制同时运行任务数量的场景中,本文将深入探讨几种有效的方法来限制C#中的并发任务数量,并通过具体的应用场景和示例代码展示如何实现这些方法,需要的朋友可以参考下
    2024-12-12
  • c#采用toml做配置文件遇到的坑

    c#采用toml做配置文件遇到的坑

    这篇文章主要介绍了c#采用toml做配置文件遇到的坑,本文通过实例代码给大家介绍的非常详细,通过本文介绍得出c#用toml文件读取非整数字请用double,不要用float,decimal倒无所谓,反正编译不过,切记不要用float,需要的朋友可以参考下
    2024-04-04
  • C# 位运算符整理

    C# 位运算符整理

    在C#中可以对整型运算对象按位进行逻辑运算。按位进行逻辑运算的意义是:依次取被运算对象的每个位,进行逻辑运算,每个位的逻辑运算结果是结果值的每个位。
    2008-10-10
  • 通过C#实现发送自定义的html格式邮件

    通过C#实现发送自定义的html格式邮件

    本篇文章主要介绍了通过C#实现发送自定义的html格式邮件,详细的介绍了发送HTML格式邮件的方法,有兴趣的可以了解一下。
    2017-02-02
  • c#完美截断字符串代码(中文+非中文)

    c#完美截断字符串代码(中文+非中文)

    c#完美截断字符串代码(中文+非中文),需要的朋友可以参考下
    2012-08-08
  • 解析c#在未出现异常情况下查看当前调用堆栈的解决方法

    解析c#在未出现异常情况下查看当前调用堆栈的解决方法

    本篇文章是对c#在未出现异常情况下查看当前调用堆栈的解决方法进行了详细的分析介绍,需要的朋友参考下
    2013-05-05

最新评论