从Java源码上复习反射的一些问题
【摘要】 Q:用反射获取到的method对象, 是返回一个method引用,还是返回1个拷贝的method对象?A:反射拿method对象时, 会做一次拷贝,而不是直接返回引用,因此最好对频繁使用的同一个method做缓存,而不是每次都去查找。Q:getMethods()后自己做遍历获取方法和getMethod(methodName) 直接获取方法, 为什么性能会有差异?A:getMethods() ...
Q:
用反射获取到的method对象, 是返回一个method引用,还是返回1个拷贝的method对象?
A:
反射拿method对象时, 会做一次拷贝,而不是直接返回引用,因此最好对频繁使用的同一个method做缓存,而不是每次都去查找。
Q:
getMethods()后自己做遍历获取方法
和getMethod(methodName) 直接获取方法, 为什么性能会有差异?
A:
getMethods() 返回method数组时,每个method都做了一次拷贝。
getMethod(methodName)只会返回那个方法的拷贝, 性能的差异就体现在拷贝上。
Q:
获取方法时,jvm内部其实有缓存,但是返回给外部时依然会做拷贝。
那么该method的缓存是持久存在的吗?
A:
不是持久存在的,内存不足时会被回收。
源码如下:
可以看到这是一个软引用。
软引用的定义:
内存紧张时可能会被回收,不过也可以通过-XX:SoftRefLRUPolicyMSPerMB参数控制回收的时机,
只要发生GC就会将其回收
如果reflectionData被回收之后,又执行了反射方法,那只能通过newReflectionData方法重新创建一个这样的对象了
Q: 反射是线程安全的吗?
A:
是线程安全的。 获取反射的数据时,通过cas去获取。
Q:
a普通方法调用
b反射方法调用
c关闭安全检查的反射方法调用,性能差异如下:
b反射方法调用和c关闭安全检查的反射方法调用的性能差异在哪?
普通方法调用和关闭安全检查的反射方法调用的性能差异在哪?
A:
-
安全检查的性能消耗在于
,SecurityManager.checkPermission(SecurityConstants.CHECK_MEMBER_ACCESS_PERMISSION); 这项检测需要运行时申请 RuntimePermission(“accessDeclaredMembers”)。
所以如果不考虑安全检查, 对反射方法调用invoke时, 应当设置 Method#setAccessible(true) -
普通方法和反射方法的性能差异在于
- Method#invoke 方法会对参数做封装和解封操作
- 需要检查方法可见性
- 需要校验参数
- 反射方法难以内联
- JIT 无法优化
【声明】本内容来自华为云开发者社区博主,不代表华为云及华为云开发者社区的观点和立场。转载时必须标注文章的来源(华为云社区)、文章链接、文章作者等基本信息,否则作者和本社区有权追究责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱:
cloudbbs@huaweicloud.com
- 点赞
- 收藏
- 关注作者
作者其他文章
评论(0)