怎样提高使用Java反射的效率?

    作者:课课家教育更新于: 2019-05-06 14:46:08

    大神带你学编程,欢迎选课

    如何提高使用java反射的效率?

     Java是当今最流行的编程技术,是Sun公司推出的Java程序设计语言和Java平台(即JavaSE, JavaEE, JavaME)的总称。Java 技术具有卓越的通用性、高效性、平台移植性和安全性,广泛应用于个人PC、数据中心、游戏控制台、科学超级计算
     
    机、移动电话和互联网,同时拥有全球最大的开发者专业社群。

    难道反射真的很慢?那跟我们平时正常创建对象调用方法比慢多少? 估计很多人都没去测试过,只是“道听途说”。下面我们就直接通过一些测试用例来直观的感受一下“反射”。

    Java_Java环境_Java安装

    前言

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

    正文

    准备测试对象

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

    1. public class TestUser {  
    2.     private Integer id;  
    3.     private String name;   
    4.     public String sayHi(){  
    5.         return "hi";  
    6.     }  
    7.     public Integer getId() {  
    8.         return id;  
    9.     }  
    10.     public void setId(Integer id) {  
    11.         this.id = id;  
    12.     }  
    13.     public String getName() {  
    14.         return name;  
    15.     }  
    16.     public void setName(String name) {  
    17.         this.name = name;  
    18.     }  

    测试创建100万个对象

    1. // 通过普通方式创建TestUser对象  
    2. @Test  
    3. public void testCommon(){  
    4.     long start = System.currentTimeMillis();  
    5.     TestUser user = null;  
    6.     int i = 0;  
    7.     while(i<1000000){  
    8.         ++i;  
    9.         user = new TestUser();  
    10.     }  
    11.     long end = System.currentTimeMillis();  
    12.     System.out.println("普通对象创建耗时:"+(end - start ) + "ms");  
    13. }  
    14. //普通对象创建耗时:10ms  
    15. // 通过反射方式创建TestUser对象  
    16. @Test  
    17. public void testReflexNoCache() throws Exception {  
    18.     long start = System.currentTimeMillis();  
    19.     TestUser user = null;  
    20.     int i = 0;  
    21.     while(i<1000000){  
    22.         ++i;  
    23.         user = (TestUser) Class.forName("ReflexDemo.TestUser").newInstance();  
    24.     }  
    25.     long end = System.currentTimeMillis();  
    26.     System.out.println("无缓存反射创建对象耗时:"+(end - start ) + "ms");  
    27. }  
    28. //无缓存反射创建对象耗时:926ms 

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

    1. // 通过缓存反射方式创建TestUser对象  
    2. @Test  
    3. public void testReflexWithCache() throws Exception {  
    4.     long start = System.currentTimeMillis();  
    5.     TestUser user = null;  
    6.     Class rUserClass = Class.forName("RefleDemo.TestUser");  
    7.     int i = 0;  
    8.     while(i<1000000){  
    9.         ++i;  
    10.         user = (TestUser) rUserClass.newInstance();  
    11.     }  
    12.     long end = System.currentTimeMillis();  
    13.     System.out.println("通过缓存反射创建对象耗时:"+(end - start ) + "ms");  
    14. }  
    15. //通过缓存反射创建对象耗时:41ms 

    咦?这种操作只需要41ms了,大大提高了反射创建对象的效率。为什么会快这么多呢?

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

    测试反射调用方法

    1. @Test  
    2. public void testReflexMethod() throws Exception {  
    3.     long start = System.currentTimeMillis();  
    4.     Class testUserClass = Class.forName("RefleDemo.TestUser");  
    5.     TestUser testUser = (TestUser) testUserClass.newInstance();  
    6.     Method method = testUserClass.getMethod("sayHi");  
    7.     int i = 0;  
    8.     while(i<100000000){  
    9.         ++i;  
    10.         method.invoke(testUser);  
    11.     }  
    12.     long end = System.currentTimeMillis();  
    13.     System.out.println("反射调用方法耗时:"+(end - start ) + "ms");  
    14. }  
    15. //反射调用方法耗时:330ms  
    1. @Test  
    2. public void testReflexMethod() throws Exception {  
    3.     long start = System.currentTimeMillis();  
    4.     Class testUserClass = Class.forName("RefleDemo.TestUser");  
    5.     TestUser testUser = (TestUser) testUserClass.newInstance();  
    6.     Method method = testUserClass.getMethod("sayHi");  
    7.     int i = 0;  
    8.     while(i<100000000){  
    9.         ++i;  
    10.         method.setAccessible(true);  
    11.         method.invoke(testUser);  
    12.     }  
    13.     long end = System.currentTimeMillis();  
    14.     System.out.println("setAccessible=true 反射调用方法耗时:"+(end - start ) + "ms");  
    15. }  
    16. //setAccessible=true 反射调用方法耗时:188ms 

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

    极致的反射

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

    这里就不介绍它的用法了,有兴趣的朋友可以直接传送过去:https://github.com/EsotericSoftware/reflectasm

    结语

    最后总结一下,为了更好的使用反射,我们应该在项目启动的时候将反射所需要的相关配置及数据加载进内存中,在运行阶段都从缓存中取这些元数据进行反射操作。大家也不用惧怕反射,虚拟机在不断的优化,只要我们方法用的对,它并没有”传

    闻“中的那么慢,当我们对性能有极致追求的时候,可以考虑通过三方包,直接对字节码进行操作。

    java的不同体系就决定了其不同的用。java有三大体系:J2SE、JEME和J2EE.J2SE主要是桌面应用程序的开发,不过这方面发展不是很好,J2ME主要是手机游戏的开发。JAVA目前应用最广泛的是J2EE(企业级java),主要是企业管理软件的开发,现

    在很多企业管理软件都是J2EE开发的,这是JAVA语言的跨平台特性决定的(这也就是大家所说的做网站,时间却高于做网站)。

课课家教育

未登录

1