Java反射影响程序执行效率吗

真的影响!!!

什么是反射

回忆下初学java时的场景:
一切皆对象!new出对象后,直接用对象点方法执行,完成一次调用!(当然也可以用类直接点方法,执行静态方法)

直到学习了jdbc,第一步就是Class.forName(String className).当时老师说,这是为了加载驱动0.0,如果不执行这一句直接获取jdbc连接,会报找不到类(后来的版本改为自动加载,不需要显示的调用Class.forName了)。这里就用到了反射。那么反射到底是什么?

有反就有正,正常情况是:已知一个类,可以通过类名调用静态方法及静态属性,或者new出对象调用实例方法及实例属性。而对于一个未知的类,通过Reflection(反射)动态解析获取类的属性和方法并使用,这就是反射。

反射是java的一个非常重要的高级特性,使用反射可以完全解析程序中的任何类并使用,使程序拥有了动态特性。

举个简单例子,sql查出来的字段如何自动填充到bean中?(此处有个前提,db字段和bean属性有规律,或者存在一个映射关系,且bean实现了属性的get/set方法)
1.sql查出来的结果集转化为map(k-v : 字段名-值)
2.通过反射解析bean获取属性(Field)及方法(Method)
3.通过map key对应到属性及方法,如 user_name,对应bean属性userName,方法setUserName
4.创建一个bean实例A(getInstance),执行方法Method 的invoke方法,传入map value,相当于执行A.setUserName(value)
这样我们就可以处理所有符合上述情况的查询,而不是在程序中写死只能处理一个具体的类,orm框架就是大量用到了反射。

反射为什么影响效率

开局一句话,剩下全靠编!
反射这么好用,真的影响效率吗,还能愉快使用吗?
使用反射真的没有直接调用快,但是该用还是要用,影响效率的原因搞清楚了,尽可能避免就是了。下面来看看,反射怎么影响效率了 。

测试反射是否影响效率,思路如下:
1.创建一个Student类,属性包含姓名、年龄、性别,方法say返回属性拼接
2.创建一个数据初始化方法,可以返回指定数量的Student集合,Student名字固定,年龄性别可以给随机值
3.创建正常遍历Student集合的方法test
4.创建反射遍历Student集合的方法testReflect,此方法在遍历开始前反射得到Student say方法的Method
5.创建反射遍历Student集合的方法testReflectOne,此方法在遍历开始后每次遍历时反射得到Student say方法的Method
6.创建反射遍历Student集合的方法testReflectPre,此方法提前反射获取Student say方法的Method,多次遍历不会重复使用反射
7.循环多次调用3/4/5/6方法,打印平均及总耗时,对比效率

代码如下:

import lombok.Data;

/**
 * <p></p>
 *
 * @author 0.0
 * @since 2020-09-27 16:52
 */
@Data
public class Student {

    private String name;

    private int age;

    private int sex;// 男/女/位置:0/1/-1

    public String say() {
        StringBuffer sb = new StringBuffer();
        sb.append("我的名字叫[").append(name).append("], 年龄[").append(age).append("], 性别[").append(this.getSexDisplay()).append("]");
        return sb.toString();
    }

    private String getSexDisplay() {
        switch (sex) {
            case 0: return "男";
            case 1: return "女";
            case -1: return "未知";
            default: return "未知";
        }
    }
}
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.List;
import java.util.Random;

/**
 * <p></p>
 *
 * @author 0.0
 * @since 2020-09-27 16:52
 */
public class MainTest {

    private static final Random RANDOM = new Random();
    // 初始化数据的总数
    private static final int DATA_INIT_TOTAL = 4000000;
    private static final List<Long> COST = new ArrayList<>();
    private static final List<Long> REFLECT_COST = new ArrayList<>();
    private static final List<Long> REFLECT_PRE_COST = new ArrayList<>();
    private static final List<Long> REFLECT_MORE_COST = new ArrayList<>();

    public static void main(String[] args) throws Exception {

        // 执行次数
        int total = 15;
        Method sayMethod = Student.class.getDeclaredMethod("say");
        List<Student> studentList = initData(DATA_INIT_TOTAL);
        for (int i = 0; i < total; i++) {
            test(i, studentList);
            testReflectPre(i, sayMethod, studentList);
            testReflect(i, studentList);
            testReflectOne(i, studentList);
            System.out.println("-----");
        }

        System.out.println("平均直接调用耗时:" + COST.stream().mapToLong(Long::longValue).sum() / COST.size() + "ms");
        System.out.println("平均预反射调用耗时:" + REFLECT_PRE_COST.stream().mapToLong(Long::longValue).sum() / COST.size() + "ms");
        System.out.println("平均反射(一次)调用耗时:" + REFLECT_COST.stream().mapToLong(Long::longValue).sum() / COST.size() + "ms");
        System.out.println("平均反射(每次)调用耗时:" + REFLECT_MORE_COST.stream().mapToLong(Long::longValue).sum() / COST.size() + "ms");
    }

    /**
     * 直接调用方法
     * @param total
     * @throws Exception
     */
    private static void test(int total, List<Student> studentList) throws Exception {
        long beginProcessTime = beginTime();
        for (Student student : studentList) {
            student.say();
        }

        long costTime = costTime(beginProcessTime);
        COST.add(costTime);
        System.out.println("第[" + total + "]直接处理总耗时:" + costTime + "ms");
    }

    /**
     * 只获取一次反射Method
     * @param total
     * @throws Exception
     */
    private static void testReflect(int total, List<Student> studentList) throws Exception {
        long beginProcessTime = beginTime();
        Method sayMethod = Student.class.getDeclaredMethod("say");
        for (Student student : studentList) {
            sayMethod.invoke(student, null).toString();
        }

        long costTime = costTime(beginProcessTime);
        REFLECT_COST.add(costTime);
        System.out.println("第[" + total + "]反射(一次)处理总耗时:" + costTime + "ms");
    }

    /**
     * 每次执行时反射获取Method
     * @param total
     * @throws Exception
     */
    private static void testReflectOne(int total, List<Student> studentList) throws Exception {
        long beginProcessTime = beginTime();
        for (Student student : studentList) {
            Method sayMethod = Student.class.getDeclaredMethod("say");
            sayMethod.invoke(student, null).toString();
        }

        long costTime = costTime(beginProcessTime);
        REFLECT_MORE_COST.add(costTime);
        System.out.println("第[" + total + "]反射(每次)处理总耗时:" + costTime + "ms");
    }

    /**
     * 预先获取反射方法,此处只执行invoke
     * @param total
     * @param sayMethod
     * @throws Exception
     */
    private static void testReflectPre(int total, Method sayMethod, List<Student> studentList) throws Exception {
        long beginProcessTime = beginTime();
        int i = 0;
        //double totalCost = 0d;
        for (Student student : studentList) {
            i++;
            long startTime = beginTime();
            sayMethod.invoke(student, null).toString();
            long totalTime = costTime(startTime);
            //totalCost += totalTime;
            /*if (i % 10000 == 0) {
                System.out.println("totalCost:" + totalCost + "ms");
                System.out.println("totalCost avg:" + String.format("%.5f", totalCost / i) + "ms");
            }*/
        }
        long costTime = costTime(beginProcessTime);
        REFLECT_PRE_COST.add(costTime);
        System.out.println("第[" + total + "]预反射处理总耗时:" + costTime + "ms");
    }

    private static long beginTime() {
        long beginTime = System.currentTimeMillis();
        return beginTime;
    }

    private static long costTime(long beginTime) {
        long endTime = System.currentTimeMillis();
        return (endTime - beginTime);
    }

    public static List<Student> initData(int size) {
        List<Student> studentList = new ArrayList<>();
        for (int i = 0; i < size; i++) {
            Student tmp = new Student();
            tmp.setName("张三");
            tmp.setAge(RANDOM.nextInt(100));
            tmp.setSex((i + RANDOM.nextInt(5)) % 2);
            studentList.add(tmp);
        }

        return studentList;
    }
}

看下执行结果:

第[0]直接处理总耗时:587ms
第[0]预反射处理总耗时:502ms
第[0]反射(一次)处理总耗时:377ms
第[0]反射(每次)处理总耗时:780ms
-----
第[1]直接处理总耗时:292ms
第[1]预反射处理总耗时:373ms
第[1]反射(一次)处理总耗时:280ms
第[1]反射(每次)处理总耗时:686ms
-----
第[2]直接处理总耗时:274ms
第[2]预反射处理总耗时:294ms
第[2]反射(一次)处理总耗时:279ms
第[2]反射(每次)处理总耗时:647ms
-----
第[3]直接处理总耗时:273ms
第[3]预反射处理总耗时:295ms
第[3]反射(一次)处理总耗时:284ms
第[3]反射(每次)处理总耗时:645ms
-----
第[4]直接处理总耗时:269ms
第[4]预反射处理总耗时:302ms
第[4]反射(一次)处理总耗时:278ms
第[4]反射(每次)处理总耗时:644ms
-----
第[5]直接处理总耗时:276ms
第[5]预反射处理总耗时:296ms
第[5]反射(一次)处理总耗时:280ms
第[5]反射(每次)处理总耗时:647ms
-----
...

再来一次

第[0]直接处理总耗时:600ms
第[0]预反射处理总耗时:480ms
第[0]反射(一次)处理总耗时:376ms
第[0]反射(每次)处理总耗时:795ms
-----
第[1]直接处理总耗时:283ms
第[1]预反射处理总耗时:374ms
第[1]反射(一次)处理总耗时:282ms
第[1]反射(每次)处理总耗时:704ms
-----
第[2]直接处理总耗时:274ms
第[2]预反射处理总耗时:299ms
第[2]反射(一次)处理总耗时:284ms
第[2]反射(每次)处理总耗时:660ms
-----
第[3]直接处理总耗时:277ms
第[3]预反射处理总耗时:298ms
第[3]反射(一次)处理总耗时:285ms
第[3]反射(每次)处理总耗时:658ms
-----
第[4]直接处理总耗时:270ms
第[4]预反射处理总耗时:303ms
第[4]反射(一次)处理总耗时:285ms
第[4]反射(每次)处理总耗时:656ms
-----
...

400W对象的遍历,能说明一定问题,反射后只是使用method.invoke方法,执行效率和直接调用差不多!,频繁使用getDeclaredMethod方法会极大地影响程序执行效率!

进入getDeclaredMethod方法,可以看到,先检查,然后查找
检查啥:“Check if client is allowed to access members. If access is denied”,方法是否允许访问。修饰符,包等
查找啥:遍历类的所有方法,根据名字和入参匹配唯一一个返回

@CallerSensitive
public Method getDeclaredMethod(String name, Class<?>... parameterTypes)
    throws NoSuchMethodException, SecurityException {
    checkMemberAccess(Member.DECLARED, Reflection.getCallerClass(), true);
    Method method = searchMethods(privateGetDeclaredMethods(false), name, parameterTypes);
    if (method == null) {
        throw new NoSuchMethodException(getName() + "." + name + argumentTypesToString(parameterTypes));
    }
    return method;
}

所以 这是一个比较复杂的操作,执行的多了,必然耗时。
但是,一个类的属性和方法是有限的,没必要每次都反射获取吧?所以,如果可能,提前反射好,后面直接用,效率影响不大

  • 4
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
1. 反射: Class 类的实例表示正在运行的 Java 应用程序中的类和接口; 枚举是一种类,注解(指的是注解Annotation)是一种接口; 每个数组都是 Class字节码类中的一个具体 对象 基本的 Java 类型(boolean、byte、char、short、int、long、float 和 double)和关键字 void 也表示为 Class 对象; 注意 : 1、 Class类 和它的实例的产生: Class的对象是已经存在的类型, 所以不能够直接new一个Class对象出来,是通过Class类中的一个方法获取到的。 例如:通过全限定路径类名 2、同一种类型不管通过什么方式得到Class的实例都是相等的;一个类型的字节码对象只有一份,在元空间。 3、Class的实例就看成是Java中我们学过的所有的数据类型在JVM中存在的一种状态(字节码对象) String.class int.class List.class int[].class 1.概念:通过一个全限定类名,获取字节码文件 2.作用: 1. 提高开发灵活度,提高程序的扩展性 2. 框架(提高开发效率的别人封装好的代码)底层都是使用反射技术。例如:Tomcat、Spring... 3. 缺点:破坏封装性,性能低下(以后,能不用反射技术就不用) 3. 使用:(重点) 1. 获取字节码文件 1. 1.1 Class clazz = Class.forName(全限定路径名) (最多使用)默认就是调用下面的方法 1.2 static 类<?> forName(String name, boolean initialize, ClassLoader loader) name:全限定路径名 initialize:表示是否初始化,默认是false loader:可以指定一个类加载器加载字节码文件 2. 全限定类名.class 3. 对象名.getClass() Class类中方法newInstance():创建当前字节码对象(只能调用无参且是public修饰的构造方法) 2. 根据字节码文件获取构造方法、普通方法、字段等 构造方法 Constructor[] constructors = clazz.getConstructors() 获取public修饰的构造方法数组 Constructor[] constructors = clazz.getDeclaredConstructors() 获取任意权限的所有造方法数组 Constructor constructor = clazz.getConstructor(Class 参数字节码)根据参数类型获取public修饰的指定的的构造方法 Constructor constructor = clazz.getDeclearConstructor(Class 参数字节码) 获取任意访问权限指定的构造方法 //通过构造方法对象去用构造方法创建对象 => 相当于new 一个对象 Object instance = constructor.newInstance(Object 实参);//可以创建任意访问权限的有参或者无参构造 普通方法 Method[] methods = clazz.getMethods() 获取public修饰的构造方法数组,有父类中的方法 Method[] methods = clazz.getDeclaredMethods() 获取任意访问权限所有造方法数组,并且都是自己的方法 Method method = clazz.getMethod(String methodName,Class... 参数字节码)根据方法名和参数类型获取指定的的方法 methodName:方法名 Class:形参类型。如果方法没有形参,则Class可变参数不用写 Method method = clazz.getDeclaredMethod(String methodName,Class... 参数字节码)根据方法名和参数类型获取指定的的方法 methodName:方法名 Class:形参类型。如果方法没有形参,则Class可变参数不用写 //通过普通方法对象调用执行方法 method.invoke(Object obj,Object... args); obj:对象。如果是对象的方法,就传入一个当前字节码创建的对象,如果是static方法,则写null args:就是具体实参 字段 Field[] fields = clazz.getFields() 获取public修饰的字段 Field[] fields = clazz.getDeclaredFields() 获取任意权限所有字段 Field field = clazz.getDeclaredField(String fieldName) 根据字段名获取任意访问权限的指定字段 Field field = clazz.Field(String fieldName)根据字段名获取public的指定字段 //通过当前的字段对象,给某一个字段赋值取值 field.get(Object obj);//如果是属于非static,就传入一个对象,如果是静态的,就传入null obj:对象 field.set(Object obj, Object value);//如果是属于非static,就传入一个对象,如果是静态的,就传入null obj:对象 value:值 String getName() 获取全限定类名(全限定,包含包名) Class类中方法 String getSimpleName() 获取类名简称 Class类中方法 Package getPackage() 获取包名 Class类中方法 T newInstance() 根据当前字节码创建对应的对象 Class类中方法 注意: 1. Class类中方法newInstance():创建当前字节码对象(只能调用无参且是public修饰的构造方法) 2. Constructor类中方法newInstance(Object 实参);//可以创建任意访问权限的有参或者无参构造 3. 私有化方法、字段、构造方法都必须破坏封装才能使用: public void setAccessible(boolean flag) true表示破坏封装,false是不破坏 是哪个private修饰的方法、字段、构造方法需要执行,就需要用这个对象破坏哪一个的封装 例如: //获取cn.itsource.reflect.User字节码文件 Class<?> clazz = Class.forName("cn.itsource.reflect.User"); //获取User的有String参构造 Constructor<?> constructor = clazz.getConstructor(String.class); //调用有参String构造,创建一个User对象 Object newInstance = constructor.newInstance("某文"); //获取private修饰的方法:testPrivate Method method2 = clazz.getDeclaredMethod("testPrivate"); method2.setAccessible(true);//破坏普通方法Method封装 //破坏封装后就可以执行了 Object invoke2 = method2.invoke(newInstance);//没有形参就不用写 System.out.println(invoke2); 4. 调用static方法、字段: 例如: Field field = clazz.getDeclaredField("country");//获取任意访问权限静态变量country field.set(null, "中国");//因为字段country是static修饰,所以不用对象调用,就传入null。字段country赋值:中文 Object object = field.get(null);//字段country取值 System.out.println(object); 2. 注解: 1.概念: 就是一个标签,有标签后,就具有某一些标签的特性。 本质就是跟类、接口、枚举平级的新结构 2.作用: 1. 帮助程序员校验代码 2. 可以提高开发效率和程序的扩展性 @Test @Before @After 3. 可以生成文档说明 api 操作步骤: 1.选中项目/代码,右键选中Export 2.输入Javadoc 3.第一个next(可以设置生成文档注释的目录),第二个next,设置字符集 如果是UTF-8编码,且有中文,请输入-encoding UTF-8 -charset UTF-8 4. 勾选一个生成完毕后,直接通过浏览器打开的选项 5. finish 3.使用:(重点) 1. 使用jdk或者别人定义好的标签 @ + 注解的名称 -- 比如@Override ->注解 2. 使用自定义的标签 1.JDK的元注解:就是专门用来声明其他注解的注解 作用:通过元注解了解其他注解的使用特点,还可以自定义注解 2.元注解: 1. @Target @Target 作用 用来限制被修饰注解的使用范围,即注解可以在类的哪些成员上使用 @Target 取值使用ElementType.() 1. CONSTRUCTOR:可以在构造器上使用注解 2. FIELD:可以在字段上使用注解 3. LOCAL_VARIABLE:可以在局部变量上使用注解 4. METHOD:可以在普通方法上使用注解 5. PACKAGE:可以在包上使用注解 6. PARAMETER:可以在参数列表上使用注解 7. TYPE:可以在类、接口(包括注解类型) 或enum上使用注解 例如:@Target(ElementType.METHOD)//意味着@Override只能在普通方法上使用 public @interface Override { } 2. @Retention

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值