我经常听说使用反射是多么糟糕。虽然我通常避免反思,也很少发现不反思就无法解决问题的情况,但我在想……
对于那些在应用程序中使用反射的人,您测量过性能影响吗?它真的那么糟糕吗?
我经常听说使用反射是多么糟糕。虽然我通常避免反思,也很少发现不反思就无法解决问题的情况,但我在想……
对于那些在应用程序中使用反射的人,您测量过性能影响吗?它真的那么糟糕吗?
当前回答
不严重。我从来没有在桌面开发中遇到过问题,除非,正如Martin所说,你在一个愚蠢的位置使用它。我听说很多人对它在桌面开发中的表现有完全不合理的担忧。
然而,在紧凑框架(我经常参与)中,它几乎是一种诅咒,在大多数情况下应该像瘟疫一样避免。我仍然可以不经常使用它,但我必须非常小心地使用它,这就不那么有趣了。:(
其他回答
它是。但这取决于你要做什么。
我使用反射动态加载程序集(插件),它的性能“惩罚”不是问题,因为操作是我在应用程序启动期间所做的事情。
然而,如果你在一系列嵌套循环中反射,每个循环都有反射调用,我想说你应该重新审视你的代码:)
对于“几次”操作,反射是完全可以接受的,您不会注意到任何延迟或问题。它是一种非常强大的机制,甚至被。net使用,所以我不明白为什么你不应该尝试一下。
在Jeff Richter的演讲The Performance of Everyday Things中,他展示了通过反射调用一个方法要比正常调用慢1000倍。
Jeff的建议:如果需要多次调用该方法,请使用一次反射来找到它,然后将其分配给委托,然后调用委托。
您甚至不得不担心. net库内部为性能关键型代码所做的反射,这已经够糟糕的了。
下面的例子在当时(2008年)是过时的,但在最近的CLR版本中早就修复了。不过,总的来说,反思仍然是一件有点昂贵的事情!
举个例子:在高性能代码中,永远不要在lock (c#) / SyncLock (VB.NET)语句中使用声明为“Object”的成员。为什么?因为CLR不能锁定值类型,这意味着它必须执行运行时反射类型检查,以查看您的对象是否实际上是值类型而不是引用类型。
和所有事情一样,这都是关于评估形势。在DotNetNuke中,有一个相当核心的组件叫做FillObject,它使用反射从数据箭头填充对象。
这是一个相当常见的场景,在MSDN上有一篇文章,使用反射将业务对象绑定到ASP。NET表单控件,涵盖了性能问题。
除了性能之外,我不喜欢在这种特定情况下使用反射的一件事是,它往往会降低快速理解代码的能力,对我来说,当你考虑到与强类型数据集或类似LINQ to SQL的数据集相比,你还失去了编译时的安全性,这似乎不值得付出努力。
反射的代价很高,因为每当您请求与参数列表匹配的方法时,运行时必须进行许多检查。在内部的某个深处,存在一些代码,它们循环遍历一个类型的所有方法,验证其可见性,检查返回类型,并检查每个参数的类型。所有这些都需要时间。
当你在内部执行这个方法时,有一些代码会在执行实际的目标方法之前检查你传递了一个兼容的参数列表。
如果可能,总是建议缓存方法句柄,如果将来要不断地重用它。像所有好的编程技巧一样,避免重复通常是有意义的。在这种情况下,不断地查找具有特定参数的方法,然后每次都执行它将是一种浪费。
戳一下源代码,看看正在做什么。