三种属性操作性能比较:PropertyInfo + Expression Tree + Delega“.NET技术”te.CreateDelegate

简介:   在《上篇》中,我比较了三种属性操作的性能:直接操作,单纯通过PropertyInfo反射和IL Emit。本篇继续讨论这个话题,我们再引入另外两种额外的属性操作方式:Expression Tree(这和IL Emit基本一致)和通过Delegate的静态方法CreateDelegate创建相应的委托进行属性的赋值和取值。

  在《上篇》中,我比较了三种属性操作的性能:直接操作,单纯通过PropertyInfo反射和IL Emit。本篇继续讨论这个话题,我们再引入另外两种额外的属性操作方式:Expression Tree(这和IL Emit基本一致)和通过Delegate的静态方法CreateDelegate创建相应的委托进行属性的赋值和取值。[源代码从这里下载]

目录
一、定义测试相关的接口、类型和委托
二、通过Expression Tree的方式创建用于属性操作的委托
三、编写属性赋值操作测试方法
四、编写属性取值操作测试方法
五、执行测试程序,查看测试结果
六、如果在Expression Tree中避免类型转换呢?

  一、定义测试相关的接口、类型和委托

  我首先定义了一个Bar类型和IFoo接口,该接口中仅仅包含一个类型和名称为Bar的可读写属性。Foo1、Foo2和Foo3均实现接口IFoo,这些接口和类型定义如下:

 
   
public class Bar{ }
public interface IFoo
{
Bar Bar {
get ; set ; }
}
public class Foo1 : IFoo
{
public Bar Bar { get ; set ; }
}
public class Foo2 : IFoo
{
public Bar Bar { get ; set ; }
}
public class Foo3 : IFoo
{
public Bar Bar { get ; set ; }
}

  然后定义如下两个委托:GetPropertyValue和SetPropertyValue。如它们的名称所表示的那些,它们分别表示属性取值和赋值操作:

 
 
public delegate Bar GetPropertyValue();
public delegate void SetPropertyValue(Bar bar);

  二、通过Expression Tree的方式创建用于属性操作的委托

  接下来我们编写Expression Tree的方式完成属性赋值和取值的操作,它们实现在如下两个静态方法中:CreateGetPropertyValueFunc和CreateSetPropertyValueAction。下面是CreateGetPropertyValueFunc的定义,它返回的是一个Func<object.object>委托:

 
 
public static Func < object , object > CreateGetPropertyValueFunc()
{
var property
= typeof (IFoo).GetProperty( " Bar " );
var target
= Expression.Parameter( typeof ( object ));
var castTarget
= Expression.Convert(target, typeof (IFoo));
var getPropertyValue
= Expression.Property(castTarget, property);
var castPropertyvalue
= Expression.Convert(getPropertyValue, typeof ( object ));
return Expression.Lambda < Func < object , object >> (castPropertyvalue , target).Compile();
}

  下面是CreateSetPropertyValueAction方法,返回一个Action<object.object>委托:

 
 
public static Action < object , object > CreateSetPropertyValueAction()
{
var property
= typeof (IFoo).GetProperty( " Bar " );
var target
= Expression.Parameter( typeof ( object ));
var propertyValue
= Expression.Parameter( typeof ( object ));
var castTarget
= Expression.Convert(target, typeof (IFoo));
var castPropertyValue
= Expression.Convert(propertyValue, property.PropertyType);
var setPropertyValue
= Expression.Call(castTarget, property.GetSetMethod(), castPropertyValue);
return Expression.Lambda < Action < object , object >> (setPropertyValue, target, propertyValue).Compile();
}

  三、编写属性赋值操作测试方法

  接下来我们编写程序测试三种不同的属性赋值操作分别具有怎样的性能,所有的测试代码定义在如下TestSetPropertyValue静态方法中。该方法参数表示进行属性赋值操作迭代的次数,每次迭代分别对Foo1、Foo2和Foo3三个对象的Bar属性进行赋值。最后打印出三种赋值操作分别的耗时,时间单位为毫秒。

 
 
public static void TestSetPropertyValue( int times)
{
var foo1
= new Foo1();
var foo2
= new Foo2();
var foo3
= new Foo3();
var bar
= new Bar();
var property
= typeof (IFoo).GetProperty( " Bar " );
var setAction
= CreateSetPropertyValueAction();
var setDelegate1
= CreateSetPropertyValueDelegate(foo1);
var setDelegate2
= CreateSetPropertyValueDelegate(foo2);
var setDelegate3
= CreateSetPropertyValueDelegate(foo3);

var stopwatch
= new Stopwatch();
stopwatch.Start();
for ( int i = 0 ; i < times; i ++ )
{
property.SetValue(foo1, bar,
null );
property.SetValue(foo2, bar,
null );
property.SetValue(foo3, bar,
null );
}
var duration1
= stopwatch.ElapsedMilliseconds;

stopwatch.Restart();
for ( int i = 0 ; i < times; i ++ )
{
setAction(foo1, bar);
setAction(foo2, bar);
setAction(foo3, bar);
}
var duration2
= stopwatch.ElapsedMilliseconds;

stopwatch.Restart();
for ( int i = 0 ; i < times; i ++ )
{
setDelegate1(bar);
setDelegate2(bar);
setDelegate3(bar);
}
var duration3
= stopwatch.ElapsedMilliseconds;
Console.WriteLine(
" {0, -15}{1,-15}{2,-15}{3,-15} " , times, duration1, duration2, duration3);
}

  四、编写属性取值操作测试方法

  属性取值操作的测试方法TestGetPropertyValue与TestSetPropertyValue结构一样。先实例化三个IFoo对象(类型分别分Foo1、Foo2和Foo3),并初始化了它们的Bar属性。然后按照三种不同的方式获取该属性值,并打印出它们各自的耗时。

 
 
public static void TestGetPropertyValue( int times)
{
var foo1
= new Foo1 { Bar = new Bar() };
var foo2
= new Foo2 { Bar = new Bar() };
var foo3
= new Foo3 { Bar = new Bar() };

var property
= typeof (IFoo).GetProperty( " Bar " );
var getFunc
= CreateGetPropertyValueFunc();
var getDelegate1
= CreateGetPropertyValueDelegate(foo1);
var getDelegate2
= CreateGetPropertyValueDelegate(foo2);
var getDelegate3
= CreateGetPropertyValueDelegate(foo3);

var stopwatch
= new Stopwatch();
stopwatch.Start();
for ( int i = 0 ; i < times; i ++ )
{
var bar1
= property.GetValue(foo1, null );
var bar2
= property.GetValue(foo2, null );
var bar3
= property.GetValue(foo3, null );
}
var duration1
= stopwatch.ElapsedMilliseconds;

stopwatch.Restart();
for ( int i = 0 ; i < times; i ++ )
{
var bar1
= getFunc(foo1);
var bar2
= getFunc(foo2);
var bar3
= getFunc(foo3);
}
var duration2
= stopwatch.ElapsedMilliseconds;

stopwatch.Restart();
for ( int i = 0 ; i < times; i ++ )
{
var bar1
= getDelegate1();
var bar2
= getDelegate2();
var bar3
= getDelegate3();
}
var duration3
= stopwatch.ElapsedMilliseconds;

Console.WriteLine(
" {0, -15}{1,-15}{2,-15}{3,-15} " , times, duration1, duration2, duration3);
}

  五、执行测试程序,查看测试结果

  我们直接通过一个Console应用来测试,在Main()方法中编写了如下的测试程序。先三次调用TestSetPropertyValue方法测试属性赋值操作,传入表示迭代次数的参数分别为10000(一万)、100000(十万)和1000000(一百万)。然后按照相同的方式调用TestGetPropertyValue测试属性取值操作。

 
 
static void Main()
{
Console.WriteLine(
" {0, -15}{1,-15}{2,-15}{3,-15} " , " Times " , " Reflection " , " Expression " , " Delegate " );
TestSetPropertyValue(
10000 );
TestSetPropertyValue(
100000 );
TestSetPropertyValue(
1000000 );

Console.WriteLine();

TestGetPropertyValue(
10000 );
TestGetPropertyValue(
100000 );
TestGetPropertyValue(
1000000 );
}

  从下面的输出结果来看,不论是属性的赋值还是取值,单纯通过PropertyInfo的方式所耗用的时间都比其它两种形式要长的多。至于其它两种(Expression Tree和通过Delegate.CreateDelegate创建委托)来说,后者又比前者有明显的优势。

 
 
Times Reflection Expression Delegate
10000 109 2 0
100000 992 21 3
1000000 9872 210 37

10000 80 2 0
100000 800 23 2
1000000 8007 239 28

  六、如果在Expression Tree中避免类型转换呢?

  当我们调用Delegate的静态方法CreateDelegate是,需要指定具体的委托类型。对于属性的操作来说,属性类型需要与指定的委托类型相匹配,所以这就避免了类型转化这个步骤。但是对于Expression Tree的属性操作来说,由于返回的类型是Func<object,object>和Action<object,object>,需要对目标对象和属性值进行两次类型转换。如果将类型转换这个步骤从Expression Tree中移掉,两者的性能是否一致呢?

  我们不妨来试试看。现在我们修改CreateGetPropertyValueFunc和CreateSetPropertyValueAction这两个静态方法,让它们直接返回Func<IFoo,Bar>和Acti上海企业网站设计与制作on<IFoo, Bar>,并去掉Expression.Convert语句。两个方法现在的定义如下:

 
 
public static Func < IFoo, Bar > CreateGetPropertyValueFunc()
{
var property
= typeof (IFoo).GetProperty( " Bar " );
var target
= Expression.Parameter( typeof (IFoo));
var getPropertyValue
= Expression.Property(target, property);
return Expression.Lambda < Func < IFoo, Bar >> (getPropertyValue, target).Compile();
}
public static Action < IFoo, Bar > CreateSetPropertyValueAction()
{
var property
= typeof (IFoo).GetProperty( " Bar " );
var target
= Expression.Parameter( typeof 上海企业网站制作"color: #000000;">(IFoo));
var propertyValue
= Expression.Parameter( typeof (Bar));
var setPropertyValue
= Expression.Call(target, property.GetSetMethod(), propertyValue);
return Expression.Lambda < Action < IFoo, Bar >> (setPropertyValue, target, propertyValue).Compile();
}

  这种情况下,再次运行我们的测试程序,你会得到如下的输出结果。从中我们不难看出,通过上面的修改,Expression Tree形式的操作在性能上得到了一定的提升,但是和第三种依然有一定的差距。

 
 
Times Reflection Expression Delegate
10000 107 1 0
100000 982 15 3
1000000 9802 157 37

10000 79 1 0
100000 789 18 2
1000000 7901 178 28
目录
相关文章
|
5天前
|
开发框架 算法 .NET
C#/.NET/.NET Core技术前沿周刊 | 第 15 期(2024年11.25-11.30)
C#/.NET/.NET Core技术前沿周刊 | 第 15 期(2024年11.25-11.30)
|
5天前
|
开发框架 Cloud Native .NET
C#/.NET/.NET Core技术前沿周刊 | 第 16 期(2024年12.01-12.08)
C#/.NET/.NET Core技术前沿周刊 | 第 16 期(2024年12.01-12.08)
|
2月前
|
自然语言处理 物联网 图形学
.NET 技术凭借其独特的优势和特性,为开发者们提供了一种高效、可靠且富有创造力的开发体验
本文深入探讨了.NET技术的独特优势及其在多个领域的应用,包括企业级应用、Web应用、桌面应用、移动应用和游戏开发。通过强大的工具集、高效的代码管理、跨平台支持及稳定的性能,.NET为开发者提供了高效、可靠的开发体验,并面对技术更新和竞争压力,不断创新发展。
76 7
|
2月前
|
开发框架 安全 .NET
在数字化时代,.NET 技术凭借跨平台兼容性、丰富的开发工具和框架、高效的性能及强大的安全稳定性,成为软件开发的重要支柱
在数字化时代,.NET 技术凭借跨平台兼容性、丰富的开发工具和框架、高效的性能及强大的安全稳定性,成为软件开发的重要支柱。它不仅加速了应用开发进程,提升了开发质量和可靠性,还促进了创新和业务发展,培养了专业人才和技术社区,为软件开发和数字化转型做出了重要贡献。
34 5
|
2月前
|
传感器 人工智能 供应链
.NET开发技术在数字化时代的创新作用,从高效的开发环境、强大的性能表现、丰富的库和框架资源等方面揭示了其关键优势。
本文深入探讨了.NET开发技术在数字化时代的创新作用,从高效的开发环境、强大的性能表现、丰富的库和框架资源等方面揭示了其关键优势。通过企业级应用、Web应用及移动应用的创新案例,展示了.NET在各领域的广泛应用和巨大潜力。展望未来,.NET将与新兴技术深度融合,拓展跨平台开发,推动云原生应用发展,持续创新。
46 4
|
2月前
|
开发框架 .NET C#
.NET 技术凭借高效开发环境、强大框架支持及跨平台特性,在软件开发中占据重要地位
.NET 技术凭借高效开发环境、强大框架支持及跨平台特性,在软件开发中占据重要地位。从企业应用到电子商务,再到移动开发,.NET 均展现出卓越性能,助力开发者提升效率与项目质量,推动行业持续发展。
36 4
|
2月前
|
机器学习/深度学习 人工智能 物联网
.NET 技术:引领未来开发潮流
.NET 技术以其跨平台兼容性、高效的开发体验、强大的性能表现和安全可靠的架构,成为引领未来开发潮流的重要力量。本文深入探讨了 .NET 的核心优势与特点,及其在企业级应用、移动开发、云计算、人工智能等领域的广泛应用,展示了其卓越的应用价值和未来发展前景。
65 5
|
2月前
|
机器学习/深度学习 人工智能 Cloud Native
在数字化时代,.NET 技术凭借其跨平台兼容性、丰富的类库和工具集以及卓越的性能与效率,成为软件开发的重要平台
在数字化时代,.NET 技术凭借其跨平台兼容性、丰富的类库和工具集以及卓越的性能与效率,成为软件开发的重要平台。本文深入解析 .NET 的核心优势,探讨其在企业级应用、Web 开发及移动应用等领域的应用案例,并展望未来在人工智能、云原生等方面的发展趋势。
45 3
|
2月前
|
敏捷开发 缓存 中间件
.NET技术的高效开发模式,涵盖面向对象编程、良好架构设计及高效代码编写与管理三大关键要素
本文深入探讨了.NET技术的高效开发模式,涵盖面向对象编程、良好架构设计及高效代码编写与管理三大关键要素,并通过企业级应用和Web应用开发的实践案例,展示了如何在实际项目中应用这些模式,旨在为开发者提供有益的参考和指导。
39 3
|
2月前
|
开发框架 安全 Java
.NET技术的独特魅力与优势,涵盖高效的开发体验、强大的性能表现、高度的可扩展性及丰富的生态系统等方面,展示了其在软件开发领域的核心竞争力
本文深入探讨了.NET技术的独特魅力与优势,涵盖高效的开发体验、强大的性能表现、高度的可扩展性及丰富的生态系统等方面,展示了其在软件开发领域的核心竞争力。.NET不仅支持跨平台开发,具备出色的安全性和稳定性,还能与多种技术无缝集成,为企业级应用提供全面支持。
37 3