反射的性能差在哪里?

反射的性能差在哪里?,第1张

参考: https://www.jianshu.com/p/4e2b49fa8ba1

下图是一亿次循环的耗时:

反射的速度差异只在大量连续使用才能明显看出来,理论上100万次才会说反射很慢,对于一个单进单出的请求来说,反射与否根本差不了多少。

事实上各大框架注解,甚至业务系统中都在使用反射,不能因为慢就不用了。

在后台Controller中序列化请求响应信息大量使用注解,高并发就意味着连续百万级别调用反射成为可能,各大MVC框架都会着手解决这个问题,优化反射。

反射核心的是getMethod和invoke了,分析下两者的耗时差距,在一亿次循环下的耗时。

证明getMethod很耗时,所以说我们要优先优化getMethod,看看为什么卡?

getMethod最后直接调用native方法,无解了。想复写优化getMethod是不可能的了,官方没毛病。

但是我们可以不需要每次都getMethod啊,我们可以缓存到redis,或者放到Spring容器中,就不需要每次都拿了。

我们看下invoke的源码:

尴尬,最后还是native方法,依然没毛病。

invoke不像getMethod可以缓存起来重复用,没法优化。

所以这里需要引入ASM,并做了个工具库reflectAsm:

参考: https://blog.csdn.net/zhuoxiuwu/article/details/78619645 , https://github.com/EsotericSoftware/reflectasm

使用如下:

参考: https://blog.csdn.net/z69183787/article/details/51657771

invoke是没办法优化的,也没办法做到像直接调用那么快。所以大佬们脑洞大开,不用反射的invoke了。原理如下:

实际上reflectAsm是有个致命漏洞的,因为要生成文件,还得load进JVM,所以reflectAsm的getMethod特别慢:

虽然getMethod很慢,但是invoke的速度是到达了直接调用的速度了。

下面是反射优化的测试样例:

java反射需要将内存中的对象进行解析,涉及到与底层c语言的交互,速度会比较慢。

java反射得到的每一个Method都有一个root,不暴漏给外部,而是每次copy一个Method。具体的反射调用逻辑是委托给MethodAccessor的,而accessor对象会在第一次invoke的时候才创建,是一种lazy init方式。而且默认Class类会cache method对象。目前MethodAccessor的实现有两种,通过设置inflation,一个native方式,一种生成java bytecode方式。native方式启动快,但运行时间长了不如java方式,个人感觉应该是java方式运行长了,jit compiler可以进行优化。所以JDK6的实现,在native方式中,有一个计数器,当调用次数达到阀值,就会转为使用java方式。默认值是15。java方式的实现,基本和非反射方式相同。

反射肯定比直接调用慢

这个毋庸置疑了,我这篇文章也不是证明反射有多高效的。

现在的快递哥很火,那我们就举个快递的例子。如果快递员就在你住的小区,那么你报一个地址:xx栋xx号,那么快递员就可以马上知道你在哪里,直接就去到你家门口;但是,如果快递员是第一次来你们这里,他是不是首先得查查百度地图,看看怎么开车过去,然后到了小区是不是得先问问物管xx栋怎么找,然后,有可能转在楼下转了两个圈才到了你的门前。

我们看上面这个场景,如果快递员不熟悉你的小区,是不是会慢点,他的时间主要花费在了查找百度地图,询问物业管理。OK,反射也是一样,因为我事先什么都不知道,所以我得花时间查询一些其他资料,然后我才能找到你。大家有兴趣可以查看反射的实现原理,以及MetaData的相关概念。

反射到底比直接调用慢多少?

好了,我们知道反射肯定慢的,那么是不是反射就不能用了呢?有些人一听到慢,就非常着急的下结论,反射怎样怎样不行,怎样怎样不能用。但是,同学,反射到底比直接调用慢多少,你造吗,能给我个实际的数据吗?很多人其实对性能只有个模糊的概念,而没有数值支撑。之前我给同事找了一个动态解析表达式的类库,他觉得不太好用,他很聪明,很快的找到了用DataTale.Compute可以实现公式的动态解析。我问他,这个方法和我给的类库性能上有什么区别?他跟我说,这个已经很快了,执行1秒都不到。我一听,就觉得不对劲,你的思想还停留在秒级,跟我谈什么性能?

怎么去判断一个函数的性能?因为函数的执行太快太快了,你需要一个放慢镜,这样才能捕捉到他的速度。怎么做?把一个函数执行一百万遍或者一千万遍,你才能真正了解一个函数的性能。也就是,你如果想判断性能,你就不能还停留在秒级,毫秒级的概念,你必须用另外一个概念替代,才能知道真正的性能。结果我同事把这两种方法执行了100w遍,确实,我提供的类库比他的快了8秒。

好了,现在拿我早两天提供的工厂方法来做测试,其中CodeTimer的实现参考赵大神的文章《一个简单的性能计数器:CodeTimer》:

测试方法如下:

代码如下

复制代码

[Test]

public void TestReflector()

{

CodeTimer.Time("Direct", 100 * 10000,

() =>

{

var instance = new ConnectionTest()

})

CodeTimer.Time("Reflect", 100 * 10000,

() =>

{

this.GetType().Assembly.CreateInstance("TestPropertyGrid.ConnectionTest")

})

}

测试结果如下:

Direct

Time Elapsed:25ms

CPU Cycles:57,582,163

Gen 0: 14

Gen 1: 0

Reflect

Time Elapsed:3,231ms

CPU Cycles:8,001,720,795

Gen 0: 269

Gen 1: 1

看到没,我们的放大镜起作用了,现在我们大概可以下这么一个结论:在执行100万遍的时候,反射大概把直接调用慢50~100倍。100倍,咋一看,是相差很大的,但是,我前文说了,别着急下结论,你要看看前提条件。自古我们就喜欢断章取义,比如“以德报怨”这个成语,好像古人说让我们遇到不好的,你不能怨恨,要更好的对待他人,别人打你左脸一巴掌,你应该把右脸伸过去让他再打一下。但实际这个成语是怎样的呢?

或曰:“以德报怨,何如?”

子曰:“何以报德?以直报怨,以德报德”

老孔的意思其实是如果别人对你好,那么你就对他好,要是他招你惹你了,你就干他娘的!你看,傻眼了吧?

有多少情况下需要考虑反射带来的影响?

我认为这个情况是非常非常少的,绝大多数的我们根本就无需考虑这个。就上我上一篇文章提到的工厂,你程序有多少个实体,有100万个吗?如果你只是在弹出窗口的时候new一下,这个百万分之十秒的影响对你很重要吗?

另外,有些人讲,我要是真有这种需求,要把一个对象new一百万遍,那不还是慢吗?这种情况有没有,有!比如我有100w条记录,需要取出来,然后通过反射赋值到一个Model类中。

但是对于这种情况,如果你真是这么想的话,我只能说,你坐办公室坐久了,脑袋生锈了,该去爬爬山,泡泡妞了。如果你需要对一个对象反射一百万遍,那么你就应该缓存这个对象了。拿我们上面那个例子来说,如果这个快递员给小区的人送一百万遍的快递还认不得路,每次都还得百度地图,然后问物业管理,你丫的你还没把他开掉了,那你脑袋不是秀逗了,要不就是任性的有钱人。

上面代码如果缓存之后执行一百万遍,跟直接调用有多大的区别?我这里就不贴代码了,免得你们直接看结果没有意思,自己把代码敲一遍,印象更深刻。

那么,还有没有更快的办法,有。比如你的快递员开始用的是IPHONE4,现在可以考虑给他买个6+。在.net中,提供了Emit的相关方法来让你更快的反射。这里送你一个通过反射快速给Model赋值的轮子“Dapper”,自己回家造去。

编程中是否应该使用反射?

其实看完上面的文字,我相信你们都有了一个初步的判断,而我的看法是:绝大多数的情况下你都可以用反射。

如果你觉得是因为反射导致你程序慢的话,那么,请先用放慢镜好好观察一下,到底是不是反射的问题。如果你确定是反射的问题,那么你再好好的考虑下是不是你没有用对反射,是不是像上面那个走了一百万遍都不认识路的快递员一样。最后,如果你觉得性能上还是不够,那么我建议你升级下硬件吧,把硬件性能上升个3%总好过你请个牛逼的工程师来帮你做这种极限的优化,有一句话我觉得很对“工程师比服务器要昂贵的多”。如果你还非得跟我较劲,那么,没办法了,你程序对性能的要求已经超出了本文讨论的范畴,如果你真有这种需求了,我觉得你也没有必要看我这篇文章了,因为你已经足够牛逼到对系统语言都有深入了解了。

大多时候,我们会把程序的性能归结于编程语言,或者使用了反射等技术,而甚少去关心自己的代码,这种心态会导致你技术的发展越来越缓慢,因为你已经失去了求知的欲望,以及一颗追求技术进步的心。请你记住,更多的时候,影响我们程序性能的,是你编程的思想,你对待编码的态度!

总结

好吧,说了这么多,估计很多人直接就拖到文章末尾然后因为文章码了这么多字而默默点了个赞,那么,我在最后给大家奉献一下本文的精华:

反射大概比直接调用慢50~100倍,但是需要你在执行100万遍的时候才会有所感觉

判断一个函数的性能,你需要把这个函数执行100万遍甚至1000万遍

如果你只是偶尔调用一下反射,请忘记反射带来的性能影响

如果你需要大量调用反射,请考虑缓存。

你的编程的思想才是限制你程序性能的最主要的因素


欢迎分享,转载请注明来源:优选云

原文地址:https://54852.com/hy/1002002.html

(0)
打赏 微信扫一扫微信扫一扫 支付宝扫一扫支付宝扫一扫
上一篇 2023-08-05
下一篇2023-08-05

随机推荐

  • 圣罗兰口红怎么分辨是不是正品

    首先,你可以从包装上辨别真伪。区别在于包装盒上的文字,其中最上面的英文字母“ROUGEPURCOUTURE”不是特别粗,只是比最下面的字母粗一点点。而且正品包装盒上的字体整体纤细舒适,假货的粗糙甚至会带一点污迹。下图是真左假右。看外包装底部

    2023-12-14
    27700
  • 资生堂悠莱官网旗舰店,资生堂悠莱适合年龄

    说到日本的护肤品,不得不说资生堂。当然资生堂旗下也有一些子品牌,比如悠莱,同样也比较受欢迎。有些女性朋友没有用过悠莱,想知道悠莱适合哪个年龄段。下面给大家详细介绍资生堂悠莱官网旗舰店,资生堂悠莱适合年龄。资生堂悠莱官网旗舰店网上是没有的,要

    2023-12-14
    24900
  • 推荐好点但不要太贵的卸隔离和B.B霜的卸妆乳!

    其次,我推荐的好的隔离霜:1、兰芝紫色隔离霜 产品卖点:营造光亮洁净肌肤,使肌肤纹理均匀细致。 推荐指数:★★★★☆ 消费者体验:很受美眉称赞的一款产品,香味清淡自然,比较好闻,质地较清爽,很容易推开,涂抹后感觉皮肤非常干净清透,最大的特点

    2023-12-14
    19300
  • 干性皮肤护肤品推荐

    干性皮肤护肤品推荐干性皮肤护肤品推荐,每个人都是一个独立的个体,因此每个人的皮肤也有所不同,有的是干性皮肤,有的则是湿性皮肤,不管是干性皮肤还是湿性皮肤,都要找到适合自己的保养皮肤的护肤品。那么来看

    2023-12-14
    24400
  • 护肤品顺序怎么用

    化妆品正确使用顺序 适合你的护肤品使用顺序:洗面奶--水(干性或混合偏干者用柔肤水,油性皮肤用爽肤水)--乳液(麻烦可以省略)--面霜。 因为你年龄还小,精华和眼霜还不需要,只要定期做些面膜就可以了。你化妆一定要卸妆啊,而且要卸干净,不

    2023-12-14
    15500
  • 摩根是何许人也他是怎么发家的

    分类:社会民生解析:华尔街之子JP摩根,摩根集团的创始人。 摩根集团的早期历史,记录了一去不返的垄断时代1871年,创始人J.P.摩根与人合伙创办德雷克塞尔—摩根公司,从事投资与信贷等银行业务1895年改名为JP.摩

    2023-12-13
    16300
  • 中国药监局药品查询

    国家药监局数据查询网址:>国家药监局药品查询网药品查询凡是药品、保健食品(即保健品)都必须经过国家食品药品监督管理局注册,并发给药品、保健品的批准文号才可以生产或上市销售,凡是经过国家食品药品监督管理局注册,并发给批准文号的药品、保健

    2023-12-13
    15600
  • 脸上长白毛是怎么回事

    分类:医疗疾病 >> 五官科问题描述:我有几个朋友脸上都有一根长长的白毛,请问是怎么回事啊?是否影响健康? 解析:听人说,那是长寿毛。呵呵,不知是不是真的 不过我以前朋友的脸上也长得有,没什么大碍。后来

    2023-12-13
    23000
  • 透明质酸的功效 什么产品中有它

    透明质酸具有良好的补水功效,是高档化妆品最好的天然保湿成分,它相容性好,几乎可以添加到任何美容化妆品中,广泛用于膏霜、乳液、化妆水、精华素、洗面奶、浴液、洗发扩发剂、摩丝、唇膏等化妆品中,一般添加量为0.05—05%。护肤成分之玻尿酸人体化

    2023-12-13
    16400

发表评论

登录后才能评论
保存