一、概述
java反射机制是在运行状态中,
对于任意一个类,都能够获得这个类所有的属性和方法;
对于任意一个对象都能调用它的属性和方法;
这种动态获取的信息以及动态调用对象的方法的功能称为java的反射机制。
反射就是将类中的成员分解为一个个java对象。
二、反射涉及的类
反射API用来生成JVM中的类、接口或则对象的信息。
★Class类:反射的核心类,可以获取类的属性,方法等信息。
★Field类:Java.lang.reflec包中的类,表示类的成员变量,可以用来获取和设置类之中的属性值。
★Method类: Java.lang.reflec包中的类,表示类的方法,它可以用来获取类中的方法信息或者执行方法。
★Constructor类: Java.lang.reflec包中的类,表示类的构造方法。
三、反射的使用
在运行期间,一个类只有一个Class对象
1、获取Class对象的三种方式
★Object --->getClass();
★任何数据类型(包括基本数据类型)都有一个“静态”的class属性
★通过Class类的静态方法,forName(String className)(常用)
2、通过反射获取构造方法并使用
★获取构造方法
1).批量的方法:
public Constructor[] getConstructors():所有"公有的"构造方法
public Constructor[] getDeclaredConstructors():获取所有的构造方法(包括私有、受保护、
默认、公有)
2).获取单个的方法,并调用:
public Constructor getConstructor(Class... parameterTypes):获取单个的"公有的"构造方法:
public Constructor getDeclaredConstructor(Class... parameterTypes):获取"某个构造方法"可以
是私有的,或受保护、默认、公有;
调用构造方法:
Constructor-->newInstance(Object... initargs)
★newInstance是 Constructor类的方法(管理构造函数的类)
newInstance(Object... initargs)
使用此 Constructor 对象表示的构造方法来创建该构造方法的声明类的新实例,并用指定的初始化参数初始化
该实例。它的返回值是T类型,所以newInstance是创建了一个构造方法的声明类的新实例对象。并为之调用
3、获取成员变量并调用
★批量的
1).Field[] getFields():获取所有的"公有字段"
2).Field[] getDeclaredFields():获取所有字段,包括:私有、受保护、默认、公有;
★获取单个的:
1).public Field getField(String fieldName):获取某个"公有的"字段;
2).public Field getDeclaredField(String fieldName):获取某个字段(可以是私有的)
★设置字段的值:
Field --> public void set(Object obj,Object value):
参数说明:
1.obj:要设置的字段所在的对象;
2.value:要为字段设置的值;
4、获取成员方法并调用
★批量的:
public Method[] getMethods():获取所有"公有方法";(包含了父类的方法也包含Object类)
public Method[] getDeclaredMethods():获取所有的成员方法,包括私有的(不包括继承的)
★获取单个的:
public Method getMethod(String name,Class<?>... parameterTypes):
参数:
name : 方法名;
Class ... : 形参的Class类型对象
public Method getDeclaredMethod(String name,Class<?>... parameterTypes)
★调用方法:
Method --> public Object invoke(Object obj,Object... args):
参数说明:
obj : 要调用方法的对象;
args:调用方式时所传递的实参;
5、通过反射运行配置文件内容
6、通过反射越过泛型检查
泛型是在编译期间起作用的。在编译后的.class文件中是没有泛型的
例如:
List<Integer> list = new ArrayList<Integer>();
list.add(100);
list.add(200);
try {
list.getClass().getMethod("add", Object.class).invoke(list, "hahhah");
} catch (IllegalAccessException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IllegalArgumentException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (InvocationTargetException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (NoSuchMethodException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (SecurityException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
for (Object obj : list) {
System.out.println(obj);
}
四、性能
泛型中涉及到的Field、Method、Constructor类其实会创建一个新对象,如果频繁的使用泛型会增加内存的开销,解决办法
是把创建的对象缓存起来。