Java反射

Java反射定义

在程序运行过程中,对于任意一个类,可以获得该类的属性和方法;对于任意一个对象,可以调用该对象的任意一个属性和方法。

运行时动态获取类的信息和动态调用对象的属性和方法称为Java反射机制。

 

反射的基石

字节码文件对象----->Class对象-----Class是Java中的一个类型

定义一个类 class A{ }----->class是Java中的关键字,用来定义类

字节码文件------ XXX.class文件

   java源文件进行编译(javac)之后的.class文件

字节码文件对象

jvm吧字节码文件加载到jvm内存中去,jvm就认定这个字节码文件是一个字节码文件对象

 

如何获取字节码文件对象

1.Object类的getClass()

2.类型.class属性

3.Class.forName("类的全路径名")【最常用】

 

获取字节码文件对象三种方式
1.通过Object类的getClass()
2.通过类型.class
3.通过Class.forName("类型名")[类型名:类的全路径名]

在一次程序的运行过程中,通过同一个类创建的字节码文件对象是同一个。

// 1.通过Object类的getClass
Person p1 = new Person();
Person p2 = new Person();

Class class1 = p1.getClass();
Class class2 = p2.getClass();

System.out.println(class1 == class2);

// 控制台打印true

// 2.通过类型.class
Class class3 = Person.class;
System.out.println(class1 == class3);

// 控制台打印true

// 3.通过Class.forName("类型名")[最常用](需要处理异常throws或者try...catch..finally)
Class class4 = Class.forName("com.study.Person");
System.out.println(class1 == class4);

// 控制台打印true

 

使用字节码文件对象
字节码文件对象中包含什么?
类 字节码文件对象
构造方法(构造器) 构造方法(构造器)对象(类型Constructor)

成员变量 成员变量对象(类型Field)

成员方法 成员方法对象(类型Method)


用字节码文件对象去构建一个类的对象
之前一直是用new 的形式去构建一个类的对象
Person p = new Person();
Person()是类的构造方法,因为对象是通过构造方法来创建。

用字节码文件中的构造方法对象来创建一个类的对象-----Constructor对象

// 获得字节码文件对象
Class clazz = Class.forName("com.study.Person");
// 得到字节码文件对象中的构造器对象
// 获取构造器对象数组(公共的构造器)
Constructor[] constructors = clazz.getConstructors();

// 获取具体的某个构造器对象
Constructor c = constructors[0];

// 调用newInstance()方法创建对象
Object person = c.newInstance();

// 便捷方式
// 直接调用clazz的newInstance()创建类对象[注意:该类必须有默认的构造方法]
Object person2 = clazz.newInstance();

构造方法对象(类型Constructor)

// 获取字节码文件对象
Class clazz = Class.forName("com.study.Person");

// 获取字节码文件对象中的构造器对象
// getConstructors()只能得到公共的构造器对象
Constructor[] constructors = clazz.getConstructors();
// getDeclaredConstructors()得到所有声明的构造器对象
Constructor[] declaredConstructors = clazz.getDeclaredConstructors();

// 得到指定的构造器对象(公共的构造器)
/*
* public Person(String name, int age)
*/
Constructor c1 = clazz.getConstructor(String.class, int.class);// 传形参
Object person1 = c1.newInstance("zhangsan", 18);// 传实参

// 得到非公共的指定构造器对象
/*
* Person(String name)
*/
Constructor c2 = clazz.getDeclaredConstructor(String.class);// 传形参
Object person2 = c2.newInstance("lisi");

/*
* private Person(int age)
*/
Constructor c3 = clazz.getDeclaredConstructor(int.class);// 传形参
// 可以得到构造器对象,但是无法使用它创建对象
// 暴力访问
c3.setAccessible(true);
Object person3 = c3.newInstance(18);

成员方法对象(类型Method)
// 获取字节码文件对象
Class clazz = Class.forName("com.study.Person");

// 得到字节码文件对象中的所有公共的Method对象
// getMethods()只能获取public修饰的方法(包含父类中的)
Method[] methods = clazz.getMethods();

System.out.println(methods.length);

for (Method m : methods) {
System.out.println(m);
}

// getDeclaredMethods() 获取所有修饰符下的方法
Method[] declaredMethods = clazz.getDeclaredMethods();

System.out.println(declaredMethods.length);

for (Method m : declaredMethods) {
System.out.println(m);
}

// 得到特定的方法 需要知道方法名、形参类型
// getMethod(方法名,形参列表) 只能得到public修饰的方法
Method method = clazz.getMethod("method", String.class);

// 调用方法
// 传统调用方法方式:对象.方法(实参);

// 创建方法所在类的对象
Object person1 = clazz.newInstance();
// 反射调用方法方式:方法.invoke(对象,实参);
Object m1 = method.invoke(person1, "张三"); // m 为返回值
System.out.println(m1);

// getMethod(方法名,形参列表) 只能得到public修饰的方法
Method declaredMethod = clazz.getDeclaredMethod("method", String.class, int.class);

// 调用方法
// 传统调用方法方式:对象.方法(实参);

// 创建方法所在类的对象
Object person2 = clazz.newInstance();
// 私有方法需要暴力访问
declaredMethod.setAccessible(true);
// 反射调用方法方式:方法.invoke(对象,实参);
Object m2 = declaredMethod.invoke(person2, "张三", 18); // m 为返回值
System.out.println(m2);

 

成员变量对象(类型Field)

// 获取字节码文件对象
Class clazz = Class.forName("com.study.Person");

// 得到所有的public修饰的成员变量
Field[] fields = clazz.getFields();

for (Field f : fields) {
System.out.println(f);
}

// 得到所有的成员变量
Field[] declaredfields = clazz.getDeclaredFields();

for (Field f : declaredfields) {
System.out.println(f);
}

// 得到指定的public修饰的成员变量对象 需要知道成员变量的名字
Field field = clazz.getField("age");
// 反射设置属性 属性名.set(对象,实参)
// 获取对象
Object person1 = clazz.newInstance();
// 设置属性值
field.set(person1, 18);

// 得到指定的成员变量对象(所有声明的) 需要知道成员变量的名字
Field declaredField = clazz.getDeclaredField("name");
// 反射设置属性 属性名.set(对象,实参)
// 获取对象
Object person2 = clazz.newInstance();
// 设置属性值
declaredField.set(person2, "zhangsan");

 

应用:

  ide(eclipse、idea),自动提示功能,对象(提示:属性、方法)

  servlet如何创建对象?  Servlet是由Tomact根据配置的类全路径信息通过Java反射机制进行创建。

posted @ 2020-08-05 22:04  努力前行,  阅读(66)  评论(0编辑  收藏  举报