反射Reflection,Java的反射是指程序在运行期可以获取一个对象的所有信息。反射是为了解决在运行期,对某个实例一无所知的情况下,如何调用其方法。
Class类
由于JVM为每个加载的class
创建了对应的Class
实例,并在实例中保存了该class
的所有信息,包括类名、包名、父类、实现的接口、所有方法、字段等,因此,如果获取了某个Class
实例,我们就可以通过这个Class
实例获取到该实例对应的class
的所有信息。
这种通过Class
实例获取class
信息的方法称为反射(Reflection)。
获取一个class
的Class
实例有三个方法:
方法一:直接通过一个class
的静态变量class
获取:
1 | Class cls = String.class; |
方法二:如果我们有一个实例变量,可以通过该实例变量提供的getClass()
方法获取:
1 | String s = "Hello"; |
方法三:如果知道一个class
的完整类名,可以通过静态方法Class.forName()
获取:
1 | Class cls = Class.forName("java.lang.String"); |
小结
JVM为每个加载的
class
及interface
创建了对应的Class
实例来保存class
及interface
的所有信息;获取一个
class
对应的Class
实例后,就可以获取该class
的所有信息;通过Class实例获取
class
信息的方法称为反射(Reflection);JVM总是动态加载
class
,可以在运行期根据条件来控制加载class。
访问字段
获取字段
Class类提供了以下几个方法来获取字段:
- Field getField(name):根据字段名获取某个public的field(包括父类)
- Field getDeclaredField(name):根据字段名获取当前类的某个field(不包括父类)
- Field[] getFields():获取所有public的field(包括父类)
- Field[] getDeclaredFields():获取当前类的所有field(不包括父类)
1 | public class Main { |
首先获取Student
的Class
实例,然后,分别获取public
字段、继承的public
字段以及private
字段,打印出的Field
类似:
1 | public int Student.score |
获取字段值
对于一个Person
实例,我们可以先拿到name
字段对应的Field
,再获取这个实例的name
字段的值:
1 | import java.lang.reflect.Field; |
正常情况下,Main
类无法访问Person
类的private
字段。要修复错误,可以将private
改为public
,或者,在调用Object value = f.get(p);
前增加f.setAccessible(true);
如果使用反射可以获取private
字段的值,那么类的封装还有什么意义?
正常情况下,通过p.name
来访问Person
的name
字段,编译器会根据public
、protected
和private
决定是否允许访问字段,这样就达到了数据封装的目的。
反射是一种非常规的用法,使用反射,首先代码非常繁琐,其次,它更多地是给工具或者底层框架来使用,目的是在不知道目标实例任何信息的情况下,获取特定字段的值。
设置字段值
通过Field实例既然可以获取到指定实例的字段值,自然也可以设置字段的值。
1 | import java.lang.reflect.Field; |
运行结果:
1 | Xiao Ming |
小结
Java的反射API提供的Field
类封装了字段的所有信息:
通过Class
实例的方法可以获取Field
实例:getField()
,getFields()
,getDeclaredField()
,getDeclaredFields()
;
通过Field实例可以获取字段信息:getName()
,getType()
,getModifiers()
;
通过Field实例可以读取或设置某个对象的字段,如果存在访问限制,要首先调用setAccessible(true)
来访问非public
字段。
通过反射读写字段是一种非常规方法,它会破坏对象的封装。
调用方法
调用方法
可以通过Class
实例获取所有Method
信息。Class
类提供了以下几个方法来获取Method
:
Method getMethod(name, Class...)
:获取某个public
的Method
(包括父类)Method getDeclaredMethod(name, Class...)
:获取当前类的某个Method
(不包括父类)Method[] getMethods()
:获取所有public
的Method
(包括父类)Method[] getDeclaredMethods()
:获取当前类的所有Method
(不包括父类)
调用构造方法
获取继承关系
动态代理
发布时间: 2019-08-07
最后更新: 2019-10-07
本文链接: https://juoyo.github.io/posts/b8b68207.html
版权声明: 本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可。转载请注明出处!