详解提高使用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 相关文章推荐
用Socket发送电子邮件(利用需要验证的SMTP服务器)
Oct 09 PHP
聊天室php&amp;mysql(六)
Oct 09 PHP
php&amp;java(二)
Oct 09 PHP
PHP4.04简明安装
Oct 09 PHP
用PHP实现读取和编写XML DOM代码
Apr 07 PHP
浅析Yii中使用RBAC的完全指南(用户角色权限控制)
Jun 20 PHP
解析csv数据导入mysql的方法
Jul 01 PHP
php根据分类合并数组的方法实例详解
Nov 06 PHP
Yii2简单实现给表单添加验证码的方法
Jul 18 PHP
php的PDO事务处理机制实例分析
Feb 16 PHP
[原创]PHP实现生成vcf vcard文件功能类定义与使用方法详解【附demo源码下载】
Sep 02 PHP
laravel 中某一字段自增、自减的例子
Oct 11 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生成excel时单元格内换行问题的解决方法
2010/08/26 PHP
PHP面向对象程序设计继承用法简单示例
2018/12/28 PHP
thinkPHP5.1框架路由::get、post请求简单用法示例
2019/05/06 PHP
PHP各种常见经典算法总结【排序、查找、翻转等】
2019/08/05 PHP
对google个性主页的拖拽效果的js的完整注释[转]
2007/04/10 Javascript
jMessageBox 基于jQuery的窗口插件
2009/12/09 Javascript
单击某一段文字改写文本颜色
2014/06/06 Javascript
AngularJS实现树形结构(ztree)菜单示例代码
2016/09/18 Javascript
JavaScript对JSON数据进行排序和搜索
2017/07/24 Javascript
详解使用angular的HttpClient搭配rxjs
2017/09/01 Javascript
基于Require.js使用方法(总结)
2017/10/26 Javascript
Vue.JS项目中5个经典Vuex插件
2017/11/28 Javascript
vue.js 实现点击展开收起动画效果
2018/07/07 Javascript
解决JS表单验证只有第一个IF起作用的问题
2018/12/04 Javascript
微信小程序实现电子签名功能
2020/07/29 Javascript
[01:01:22]VGJ.S vs OG 2018国际邀请赛淘汰赛BO3 第一场 8.22
2018/08/23 DOTA
python生成随机验证码(中文验证码)示例
2014/04/03 Python
Python类方法__init__和__del__构造、析构过程分析
2015/03/06 Python
Python爬虫模拟登录带验证码网站
2016/01/22 Python
python自动翻译实现方法
2016/05/28 Python
Python英文文本分词(无空格)模块wordninja的使用实例
2019/02/20 Python
python RC4加密操作示例【测试可用】
2019/09/26 Python
python读取hdfs并返回dataframe教程
2020/06/05 Python
使用Python绘制台风轨迹图的示例代码
2020/09/21 Python
Django使用django-simple-captcha做验证码的实现示例
2021/01/07 Python
Jupyter Notebook 远程访问配置详解
2021/01/11 Python
Python实现简单的2048小游戏
2021/03/01 Python
英国的一家创新礼品和小工具零售商:Menkind
2019/08/24 全球购物
荷兰家电购物网站:Expert.nl
2020/01/18 全球购物
简历中自我评价范文3则
2013/12/14 职场文书
3的组成教学反思
2014/04/30 职场文书
2014年合同管理工作总结
2014/12/02 职场文书
捐助倡议书
2015/01/19 职场文书
贫民窟的百万富翁观后感
2015/06/09 职场文书
热爱劳动主题班会
2015/08/14 职场文书
本地通过nginx配置反向代理的全过程记录
2021/03/31 Servers