摘要:Java反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的。..
Java反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。
反射就是把java类中的各种成分映射成一个个的Java对象。
例如:一个类有:成员变量、方法、构造方法、包等等信息,利用反射技术可以对一个类进行解剖,把个个组成部分映射成一个个对象。
(其实:一个类中这些成员方法、构造方法、在加入类中都有一个类来描述)
一、Java反射机制
1.1 反射
1 | Q: 调用类对象。class 和 forName(类名)的区别? |
1 | Class<A> classA = A.class; |
A: 仅使用。class不能进行第一次静态初始化, forname函数则可以
例如B是A的基类,下面这段代码如何? 假设有父子2个类,如下:
1 | static class Parent { } |
A: 可以比较,且返回true。
1 | Q: 用getClass并用== 可以和父类比较吗,且会返回true吗,下面这样: 注意A是B的子类。 |
1 | if (son.getClass() == Parent.class){ |
}
1 |
|
A: 可以比较,正常编译, 但是会返回false,即不相等!
Q: getDeclaredXXX 有哪几种? A: 5种:
注解Annotation
内部类Classed
构造方法Construcotor
字段Field
方法Method
1 | Q:getMethods()返回哪些方法, getDeclaredMethods()会返回哪些方法? |
其他getXXX和getDeclaredXXX的区别同理。
拿到Filed、Method、Constructor之后咋用
Method可以invoke(object, args)
1 | Constructor可以newInstance(Object…)来做构造调用。 |
1 | Q: 反射拿到Method对象后, 该对象。getModifiers() 是干嘛的? A: 返回该方法的修饰符,并且是1个整数。 |
Q: 下面这段代码会发生什么?
package com.huawei.test
1 | public class A { |
}
1 | public static void main(String[] args) { |
}
1 |
|
Q:如何提高反射的效率? A:
使用高性能反射包,例如ReflectASM
缓存反射的对,避免每次都要重复去字节码中获取。(缓存!缓存!)
1 | method反射可设置method.setAccessible(true)来关闭安全检查。 |
Q: 用反射获取到的method对象, 是返回一个method引用,还是返回1个拷贝的method对象? A: 反射拿method对象时, 会做一次拷贝,而不是直接返回引用,因此最好对频繁使用的同一个method做缓存,而不是每次都去查找。
1 | Q: getMethods()后自己做遍历获取方法,和getMethod(methodName) 直接获取方法, 为什么性能会有差异? A: getMethods() 返回method数组时,每个method都做了一次拷贝。 getMethod(methodName)只会返回那个方法的拷贝, 性能的差异就体现在拷贝上。 |
1 | private Class.ReflectionData<T> reflectionData() { |
}
可以看到这是一个软引用。
软引用的定义:内存紧张时可能会被回收,不过也可以通过-XX:SoftRefLRUPolicyMSPerMB参数控制回收的时机,只要发生GC就会将其回收。
如果reflectionData被回收之后,又执行了反射方法,那只能通过newReflectionData方法重新创建一个这样的对象了。
Q: 反射是线程安全的吗? A: 是线程安全的。 获取反射的数据时,通过cas去获取。 cas概念可以见多线程一节。
Q: a普通方法调用 b反射方法调用 c关闭安全检查的反射方法调用,性能差异如下:
b反射方法调用和c关闭安全检查的反射方法调用的性能差异在哪?普通方法调用和关闭安全检查的反射方法调用的性能差异在哪? A:
安全检查的性能消耗在于 ,SecurityManager.checkPermission(SecurityConstants.CHECK_MEMBER_ACCESS_PERMISSION); 这项检测需要运行时申请RuntimePermission(“accessDeclaredMembers”)。 所以如果不考虑安全检查, 对反射方法调用invoke时, 应当设置 Method#setAccessible(true)
普通方法和反射方法的能差异在于
Method#invoke 方法会对参数做封装和解封操作
需要检查方法可见性
需要校验参数
反射方法难以内联
JIT 无法优化
本文标题: Java程序员都要懂得
发布时间: 2019年04月15日 00:00
最后更新: 2025年12月30日 08:54
原始链接: https://haoxiang.eu.org/4fad1fb6/
版权声明: 本文著作权归作者所有,均采用CC BY-NC-SA 4.0许可协议,转载请注明出处!

