当前位置:网站首页>isAccessible()方法:使用反射技巧让你的性能提升数倍
isAccessible()方法:使用反射技巧让你的性能提升数倍
2022-07-29 05:21:00 【代码与思维】
今天这篇文章从源码的角度分析一下 isAccessible() 方法的作用,为什么将 Accessible 设置为 true 可以提升性能,在开始分析之前,我们先写一段代码。
- 声明一个普通类,里面有个
public方法getName()和private方法getAddress()
class Person {
public fun getName(): String {
return "I am DHL"
}
private fun getAddress(): String {
return "BJ"
}
}
- 通过反射获取
getName()和getAddress()方法,花 3 秒钟思考一下,下面的代码输出的结果
// public 方法
val method1 = Person::class.declaredFunctions.find { it.name == "getName" }
println("access = ${method1?.isAccessible}")
// private 方法
val method2 = Person::class.declaredFunctions.find { it.name == "getAddress" }
println("access = ${method2?.isAccessible}")
无论是调用 public getName() 方法还是调用 private getAddress() 方法,最后输出的结果都为 false,通过这个例子也间接说明了 isAccessible() 方法并不是用来表示访问权限的。
当我们通过反射调用 private 方法时,都需要执行 setAccessible() 方法设置为 true, 否者会抛出下面的异常。
java.lang.IllegalAccessException: can not access a member of class com.hi.dhl.demo.reflect.Person
如果通过反射调用 public 方法,不设置 Accessible 为 true,也可以正常调用,所以有很多小伙伴认为 isAccessible() 方法用来表示访问权限,其实这种理解是错误的。
我们一起来看一下源码是如何解释的,方法 isAccessible() 位于 AccessibleObject 类中。
public class AccessibleObject implements AnnotatedElement {
......
// NOTE: for security purposes, this field must not be visible
boolean override;
public boolean isAccessible() {
return override;
}
public void setAccessible(boolean flag) throws SecurityException {
......
}
......
}
AccessibleObject 是 Field 、 Method 、 Constructor 的父类,调用 isAccessible() 返回 override 的值,而字段 override 主要判断是否要进行安全检查。
字段 override 在 AccessibleObject 子类当中使用,所以我们一起来看一下它的子类 Method。
public Object invoke(Object obj, Object... args){
// 是否要进行安全检查
if (!override) {
// 进行快速验证是否是 Public 方法
if (!Reflection.quickCheckMemberAccess(clazz, modifiers)) {
// 返回调用这个方法的 Class
Class<?> caller = Reflection.getCallerClass();
// 做权限访问的校验,缓存调用这个方法的 Class,避免下次在做检查
checkAccess(caller, clazz, obj, modifiers);
}
}
......
return ma.invoke(obj, args);
}
字段 override 提供给子类去重写,它的值决定了是否要进行安全检查,如果要进行安全检查,则会执行 quickCheckMemberAccess() 快速验证是否是 Public 方法,避免调用 getCallerClass()。
- 如果是
Public方法,避免做安全检查,所以我们在代码中不调用setAccessible(true)方法,也不会抛出异常 - 如果不是
Public方法则会调用getCallerClass()获取调用这个方法的 Class,执行checkAccess()方法进行安全检查。
// it is necessary to perform somewhat expensive security checks.
// A more complicated security check cache is needed for Method and Field
// The cache can be either null (empty cache)
volatile Object securityCheckCache; // 缓存调用这个方法的 Class
void checkAccess(Class<?> caller, Class<?> clazz, Object obj, int modifiers){
......
Object cache = securityCheckCache; // read volatile
if(cache == 调用这个方法的 Class){
return; // ACCESS IS OK
}
slowCheckMemberAccess(caller, clazz, obj, modifiers, targetClass);
......
}
void slowCheckMemberAccess(Class<?> caller, Class<?> clazz, Object obj, int modifiers,Class<?> targetClass){
Reflection.ensureMemberAccess(caller, clazz, obj, modifiers);
Object cache = 调用这个方法的 Class
securityCheckCache = cache; // 缓存调用这个方法的 Class
}
源码中注释也说明了,如果要进行安全检查那么它的代价是非常昂贵的,所以用变量 securityCheckCache 缓存调用这个方法的 Class。如果下次使用相同的 Class,就不需要在做安全检查,但是这个缓存有个缺陷,如果换一个调用这个方法的 Class,需要再次做安全检查,并且会覆盖之前的缓存结果。
如果要在运行时修改属性或者调用某个方法时,都要进行安全检查,而安全检查是非常消耗资源的,所以 JDK 提供了一个 setAccessible() 方法,可以绕过安全检查,让开发者自己来决定是否要避开安全检查。
因为反射本身是非常慢的,如果能够避免安全检查,可以进一步提升性能,针对不同场景,分别测试了反射前后以及关闭安全检查的耗时。
| 正常调用 | 反射 | 反射优化后 | 反射优化后关掉安全检查 | |
|---|---|---|---|---|
| 创建对象 | 0.578 ms/op | 4.710 ms/op | 1.018 ms/op | 0.943 ms/op |
| 方法调用 | 0.422 ms/op | 10.533 ms/op | 0.844 ms/op | 0.687 ms/op |
| 属性调用 | 0.241 ms/op | 12.432 ms/op | 1.362 ms/op | 1.202 ms/op |
| 伴生对象 | 0.470 ms/op | 5.661 ms/op | 0.840 ms/op | 0.702 ms/op |
从测试结果可以看出来,执行 setAccessible() 方法,设置为 true 关掉安全检查之后,反射速度得到了进一步的提升,更接近于正常调用。
作者:程序员DHL
链接:https://juejin.cn/post/7121901090332737572
边栏推荐
- “山东大学移动互联网开发技术教学网站建设”项目实训日志六
- 加密资产熊市之下,PlatoFarm的策略玩法依旧能获得稳定收益
- Extreme deflation and perpetual motion machine model will promote the outbreak of platofarm
- 以‘智’提‘质|金融影像平台解决方案
- 钉钉告警脚本
- Basic use of array -- traverse the circular array to find the maximum value, minimum value, maximum subscript and minimum subscript of the array
- Use of xtrabackup
- 在uni-app项目中,如何实现微信小程序openid的获取
- mysql 的show profiles 使用。
- 机器学习让文字识别更简单:Kotlin+MVVM+华为ML Kit
猜你喜欢

Synchronous development with open source projects & codereview & pull request & Fork how to pull the original warehouse

C# 判断用户是手机访问还是电脑访问

How to survive in the bear market of encryption market?

Windos下安装pyspider报错:Please specify --curl-dir=/path/to/built/libcurl解决办法

XDFS&空天院HPC集群典型案例

并发编程学习笔记 之 ReentrantLock实现原理的探究

重庆大道云行作为软件产业代表受邀参加渝中区重点项目签约仪式

如何零代码制作深度学习的趣味app(适合新手)

PHP write a diaper to buy the lowest price in the whole network

熊市慢慢,Bit.Store提供稳定Staking产品助你穿越牛熊
随机推荐
识变!应变!求变!
完全去中心化的编程模式,不需要服务器,也不需要ip,就像一张漫无目的的网络、四处延伸
并发编程学习笔记 之 Lock锁及其实现类ReentrantLock、ReentrantReadWriteLock和StampedLock的基本用法
Power BI Report Server 自定义身份验证
裸金属云FASS高性能弹性块存储解决方案
Idea using JDBC to connect mysql database personal detailed tutorial
数组的基础使用--遍历循环数组求出数组最大值,最小值以及最大值下标,最小值下标
学习、研究编程之道
IDEA中设置自动build-改动代码,不用重启工程,刷新页面即可
全闪分布式,如何深度性能POC?
datax安装
重庆大道云行作为软件产业代表受邀参加渝中区重点项目签约仪式
熊市慢慢,Bit.Store提供稳定Staking产品助你穿越牛熊
Laravel service container (inheritance and events)
Use of file upload (2) -- upload to Alibaba cloud OSS file server
Thinkphp6 pipeline mode pipeline use
与张小姐的春夏秋冬(3)
XDFS&中国日报社在线协同编辑平台典型案例
H5 semantic label
PHP write a diaper to buy the lowest price in the whole network