详解提高使用Java反射的效率方法


Posted in PHP onApril 29, 2019

在我们平时的工作或者面试中,都会经常遇到“反射”这个知识点,通过“反射”我们可以动态的获取到对象的信息以及灵活的调用对象方法等,但是在使用的同时又伴随着另一种声音的出现,那就是“反射”很慢,要少用。难道反射真的很慢?那跟我们平时正常创建对象调用方法比慢多少? 估计很多人都没去测试过,只是”道听途说“。下面我们就直接通过一些测试用例来直观的感受一下”反射“。
正文

准备测试对象

下面先定义一个测试的类TestUser,只有id跟name属性,以及它们的getter/setter方法,另外还有一个自定义的sayHi方法。

public class TestUser { private Integer id; private String name; 

 public String sayHi(){  return "hi";

 } public Integer getId() {  return id;

 } public void setId(Integer id) {  this.id = id;

 } public String getName() {  return name;

 } public void setName(String name) {  this.name = name;

 }

}

测试创建100万个对象

// 通过普通方式创建TestUser对象@Testpublic void testCommon(){ long start = System.currentTimeMillis();

 TestUser user = null; int i = 0; while(i<1000000){

  ++i;

  user = new TestUser();

 } long end = System.currentTimeMillis();

 System.out.println("普通对象创建耗时:"+(end - start ) + "ms");

}//普通对象创建耗时:10ms
// 通过反射方式创建TestUser对象@Testpublic void testReflexNoCache() throws Exception { long start = System.currentTimeMillis();

 TestUser user = null; int i = 0; while(i<1000000){

  ++i;

  user = (TestUser) Class.forName("ReflexDemo.TestUser").newInstance();

 } long end = System.currentTimeMillis();

 System.out.println("无缓存反射创建对象耗时:"+(end - start ) + "ms");

}//无缓存反射创建对象耗时:926ms

在上面这两个测试方法中,笔者各自测了5次,把他们消耗的时间取了一个平均值,在输出结果中可以看到一个是10ms,一个是926ms,在创建100W个对象的情况下,反射居然慢了90倍左右。wtf?差距居然这么大?难道反射真的这么慢?下面笔者换一种反射的姿势,继续测试一下,看看结果如何

// 通过缓存反射方式创建TestUser对象@Testpublic void testReflexWithCache() throws Exception { long start = System.currentTimeMillis();

 TestUser user = null;

 Class rUserClass = Class.forName("RefleDemo.TestUser"); int i = 0; while(i<1000000){

  ++i;

  user = (TestUser) rUserClass.newInstance();

 } long end = System.currentTimeMillis();

 System.out.println("通过缓存反射创建对象耗时:"+(end - start ) + "ms");

}//通过缓存反射创建对象耗时:41ms

其实通过代码我们可以发现,是Class.forName这个方法比较耗时,它实际上调用了一个本地方法,通过这个方法来要求JVM查找并加载指定的类。所以我们在项目中使用的时候,可以把Class.forName返回的Class对象缓存起来,下一次使用的时候直接从缓存里面获取,这样就极大的提高了获取Class的效率。同理,在我们获取Constructor、Method等对象的时候也可以缓存起来使用,避免每次使用时再来耗费时间创建。

测试反射调用方法

@Testpublic void testReflexMethod() throws Exception { long start = System.currentTimeMillis();

 Class testUserClass = Class.forName("RefleDemo.TestUser");

 TestUser testUser = (TestUser) testUserClass.newInstance();

 Method method = testUserClass.getMethod("sayHi"); int i = 0; while(i<100000000){

  ++i;

  method.invoke(testUser);

 } long end = System.currentTimeMillis();

 System.out.println("反射调用方法耗时:"+(end - start ) + "ms");

}//反射调用方法耗时:330ms
@Testpublic void testReflexMethod() throws Exception { long start = System.currentTimeMillis();

 Class testUserClass = Class.forName("RefleDemo.TestUser");

 TestUser testUser = (TestUser) testUserClass.newInstance();

 Method method = testUserClass.getMethod("sayHi"); int i = 0; while(i<100000000){

  ++i;

  method.setAccessible(true);

  method.invoke(testUser);

 } long end = System.currentTimeMillis();

 System.out.println("setAccessible=true 反射调用方法耗时:"+(end - start ) + "ms");

}//setAccessible=true 反射调用方法耗时:188ms

这里我们反射调用sayHi方法1亿次,在调用了method.setAccessible(true)后,发现快了将近一半。查看API可以了解到,jdk在设置获取字段,调用方法的时候会执行安全访问检查,而此类操作会比较耗时,所以通过setAccessible(true)的方式可以关闭安全检查,从而提升反射效率。

极致的反射

除了上面的手段,还有没有什么办法可以更极致的使用反射呢?这里介绍一个高性能反射工具包ReflectASM。它是通过字节码生成的方式来实现的反射机制,下面是一个跟java反射的性能比较。

详解提高使用Java反射的效率方法

结语

最后总结一下,为了更好的使用反射,我们应该在项目启动的时候将反射所需要的相关配置及数据加载进内存中,在运行阶段都从缓存中取这些元数据进行反射操作。大家也不用惧怕反射,虚拟机在不断的优化,只要我们方法用的对,它并没有”传闻“中的那么慢,当我们对性能有极致追求的时候,可以考虑通过三方包,直接对字节码进行操作。

PHP 相关文章推荐
PHP在字符断点处截断文字的实现代码
Apr 21 PHP
解析thinkphp中的导入文件标签
Jun 20 PHP
解析func_num_args与func_get_args函数的使用
Jun 24 PHP
php教程之魔术方法的使用示例(php魔术函数)
Feb 12 PHP
php去除字符串换行符示例分享
Feb 13 PHP
Zend Framework 2.0事件管理器(The EventManager)入门教程
Aug 11 PHP
php使用Cookie控制访问授权的方法
Jan 21 PHP
php获取、检查类名、函数名、方法名的函数方法
Jun 25 PHP
PHP+Mysql+jQuery实现发布微博程序 php篇
Oct 15 PHP
php+ajax登录跳转登录实现思路
Jul 31 PHP
PHP简单实现模拟登陆功能示例
Sep 15 PHP
Laravel框架使用Seeder实现自动填充数据功能
Jun 13 PHP
Thinkphp整合阿里云OSS图片上传实例代码
Apr 28 #PHP
详解php中生成标准uuid(guid)的方法
Apr 28 #PHP
PHP写API输出的时用echo的原因详解
Apr 28 #PHP
PHP+mysql防止SQL注入的方法小结
Apr 27 #PHP
PHP常用工具函数小结【移除XSS攻击、UTF8与GBK编码转换等】
Apr 27 #PHP
PHP操作路由器实现方法示例
Apr 27 #PHP
PHP切割汉字的常用方法实例总结
Apr 27 #PHP
You might like
PHP中include/require/include_once/require_once使用心得
2016/08/28 PHP
PHP基于mcript扩展实现对称加密功能示例
2019/02/21 PHP
DOMAssitant最新版 DOMAssistant 2.5发布
2007/12/25 Javascript
学习JavaScript的最佳方法分享
2011/10/21 Javascript
IE 下Enter提交表单存在重复提交问题的解决方法
2014/05/04 Javascript
JS中使用apply、bind实现为函数或者类传入动态个数的参数
2016/04/26 Javascript
有关easyui-layout中的收缩层无法显示标题的解决办法
2016/05/10 Javascript
关于JS中二维数组的声明方法
2016/09/24 Javascript
详解在express站点中使用ejs模板引擎
2017/09/21 Javascript
bootstrap时间控件daterangepicker使用方法及各种小bug修复
2017/10/25 Javascript
详解 微信小程序开发框架(MINA)
2019/05/17 Javascript
解决vuex刷新状态初始化的方法实现
2019/08/15 Javascript
Vue-CLI 3 scp2自动部署项目至服务器的方法
2020/07/24 Javascript
[08:47]2018国际邀请赛 OG战队举杯时刻
2018/08/29 DOTA
python实现telnet客户端的方法
2015/04/15 Python
Python实现的递归神经网络简单示例
2017/08/11 Python
Django视图之ORM数据库查询操作API的实例
2017/10/27 Python
Django中间件基础用法详解
2019/07/18 Python
pycharm中显示CSS提示的知识点总结
2019/07/29 Python
Python2和3字符编码的区别知识点整理
2019/08/08 Python
TFRecord格式存储数据与队列读取实例
2020/01/21 Python
python_mask_array的用法
2020/02/18 Python
python+gdal+遥感图像拼接(mosaic)的实例
2020/03/10 Python
Scrapy 配置动态代理IP的实现
2020/09/28 Python
python处理写入数据代码讲解
2020/10/22 Python
Python下使用Trackbar实现绘图板
2020/10/27 Python
python matlab库简单用法讲解
2020/12/31 Python
印度最大的网上花店:Ferns N Petals(鲜花、礼品和蛋糕)
2017/10/16 全球购物
CK巴西官方网站:Calvin Klein巴西
2019/07/19 全球购物
Lucene推荐的分页方式是什么?
2015/12/07 面试题
兼职业务员岗位职责
2014/01/01 职场文书
九年级化学教学反思
2014/01/28 职场文书
2019学生会干事辞职信
2019/06/27 职场文书
如何用JavaScript实现一个数组惰性求值库
2021/05/05 Javascript
MySQL 常见的数据表设计误区汇总
2021/06/07 MySQL
详解Mysql数据库平滑扩容解决高并发和大数据量问题
2022/05/25 MySQL