反射技术可以动态创建实例,即在不知道具体类的情况创建对象,要创建对象就要获取描述对象的字节码对象,而要获取字节码对象要先获取描述字节码对象的Class
获取Class对象方式:
public static void getClassDemo_1() {
Person p=new Person();
Class clazz=p.getClass();
System.out.println(clazz);
}
public static void getClassDemo_2() {
Class clazz=Person.class;
System.out.println(clazz);
}
public static void getClassDemo_3() throws ClassNotFoundException {
String classname="com.yan.reflect.Person";
Class clazz=Class.forName(classname);//根据名字去classpath路径下找到对应的字节码文件加载进内存
System.out.println(clazz);
}
前面两种明显不符合,因为这两种前提条件是一直具体的类,故一般采用第三种
动态创建对象实例:
public static void getObject() throws ClassNotFoundException, InstantiationException, IllegalAccessException {
String className="com.yan.reflect.Person";
Class clazz=Class.forName(className);
Object o=clazz.newInstance();
}
它的调用过程与new一样,默认调用无参的构造函数,想调用有参的构造函数:
public static void getObjectByConstructor() throws Exception {
String className="com.yan.reflect.Person";
Class clazz=Class.forName(className);
Constructor cons=clazz.getConstructor(String.class,int.class);//获取构造函数对象
Object o=cons.newInstance("lisi",18);
}
Person类
package com.yan.reflect;
public class Person {
private String name;
private int age;
public Person() {
super();
}
public Person(String name,int age){
super();
this.name=name;
this.age=age;
}
}
其实不但可以获取构造函数,有了Class对象,就可以对它进行解剖,获取各种所需要的,比如字段
public static void getFieldDemo() throws Exception{
String className="com.yan.reflect.Person";
Class clazz=Class.forName(className);
Field filed=clazz.getField("age");
System.out.println(filed);
}
结果:
原因是getField只能获取到公有成员,想获取私有的成员,要用getDeclareField()