文章目录
前言
1. 动态语言的概念
动态语言是指程序在运行时可以改变其结构:新的函数可以引进,已有的函数可以被删除等结构上的变化。
比如常见的JavaScript就是动态语言,除此之外Ruby、Python等也属于动态语言,而C、C++则不属于动态语言。
从反射角度来说Java属于半动态语言。
2. 反射机制概念
在Java中的反射机制是指在运行状态中,对于任意一个类都能够知道这个类所有的属性和方法,并且对于任意一个对象,都能够调用它的任意一个方法,这种动态获取信息以及动态调用对象方法的功能称为Java的反射机制。
3. 反射的应用场合
3.1. 编译时类型和运行时类型
在Java程序中许多对象在运行时都会出现两种类型:编译时类型和运行时类型。
编译时的类型由声明对象时使用的类型来决定,运行时类型由实际赋值给对象的类型决定。如:Person p = new Student();
其中编译时类型为Person,运行时类型为Student。
3.2. 编译时类型无法获取具体方法
程序在运行时还可能接收到外部传入的对象,该对象的编译时类型为Object,但是程序有需要调用该对象的运行时类型的方法。为了解决这些问题,程序需要在运行时发现对象和类的真实信息。然而,如果编译时根本无法预知该对象和类属于哪些类,程序只能依靠运行时信息来发现该对象和类的真实信息,此时就必须使用反射。
Java反射API
1. 相关类
1.1. Class类
反射的核心类,可以获取类的属性、方法等信息
- 获得类相关的方法
方法 | 用途 |
---|---|
asSubclass(Class clazz) | 把传递的类的对象转换成代表其子类的对象 |
Cast | 把对象转换成代表类或是接口的对象 |
getClassLoader() | 获得类的加载器 |
getClasses() | 返回一个数组,数组中包含该类中所有公共类和接口类的对象 |
getDeclaredClassed() | 返回一个数组,数组中包含该类中所有类和接口类的对象 |
forName(String className) | 根据类名返回类的对象 |
getName() | 获得类的完整路径名字 |
newInstance() | 创建类的实例 |
getPackage() | 获取类的包 |
getSimpleName() | 获取类的名字 |
getSuperclass() | 获得当前类继承的父类的名字 |
getInterfaces() | 获得当前类实现的类或是接口 |
getField(String name) | 获得某个共有的数据对象 |
- 获得类中属性相关的方法
方法 | 用途 |
---|---|
getFields() | 获得所有公有的属性对象 |
getDeclaredField(String name) | 获得某个属性对象 |
getDeclaredFields() | 获得所有属性对象 |
- 获得类中构造器相关的方法
方法 | 用途 |
---|---|
getConstructor(Class …<?> parameterTypes) | 获得该类中与参数类型匹配的共有构造方法 |
getConstructors() | 获得该类的所有公有构造方法 |
getDeclaredConstructor(Class…<?> parameterTypes | 获得该类中与参数类型匹配的构造方法 |
getDeclaredConstructors() | 获得该类所有构造方法 |
- 获得类中方法相关的方法
方法 | 用途 |
---|---|
getMethod(String name, Class…<?> parameterTypes | 获得该类某个公有的方法 |
getMethods() | 获得该类所有公有的方法 |
getDeclaredMethod(String name, Class…<?> parameterTypes) | 获得该类某个方法 |
getDeclaredMethods() | 获得该类所有方法 |
- 还有其它许多方法此处省略… 想了解的朋友可以去看一下源码
1.2. Field类
java.lang.reflec包中的类,表示类的成员变量,可以用来获取和设置类之中的属性值
方法 | 用途 |
---|---|
equals(Object obj) | 属性与对象相等则返回true |
get(Object obj) | 获得对象中对应的属性值 |
set(Object obj, Object value) | 设置对象中对应的属性值 |
… | … |
1.3. Method类
java.lang.reflec包中的类,表示类的方法,它可以用来获取类中的方法信息或者执行方法
方法 | 用途 |
---|---|
invoke(Object obj, Object … args) | 传递obj对象及参数调用该对象对应的方法 |
getParameterCount | 获得方法参数数量 |
… | … |
1.4. Constructor类
java.lang.reflec包中的类,表示类的构造方法
方法 | 用途 |
---|---|
newInstance(Object … iniitargs) | 根据传递的参数创建类的对象 |
getParameterCount | 获得方法参数数量 |
… | … |
2. 反射使用步骤
- 获取想要操作类的Class对象,它是反射的核心,通过Class对象我们可以任意调用类的方法
- 调用Class类中的方法,就是反射的使用阶段
- 使用反射API来操作这些信息
3. 获取Class对象的3种方法
//调用某个对象getClass()方法
Student s = new Student();
Class clazz = s.getClass();
//调用某个类的class属性来获取该类的对应的Class对象
Class clazz = Student.class;
//使用Class类中的forName()静态方法
Class clazz = Class.forName("类的全路径");
当我们获得了想要操作的类的Class对象后,可以通过Class类种的方法获取并查看该类中的方法和属性。
假设我此时有一个Student类:
package com.soul.fanshe;
/**
* @Title: Student.java
* @Description: TODO(描述)
* @author Soul
* @date 2020-05-23 11:10:31
*/
public class Student {
private String name;
private String hight;
private Integer age;
public Student() { super(); }
public Student(String name, String hight, Integer age) {
super();
this.name = name;
this.hight = hight;
this.age = age;
}
public String getName() { return name; }
public void setName(String name) {this.name = name;}
public String getHight() { return hight; }
public void setHight(String hight) {this.hight = hight;}
public Integer getAge() { return age;}
public void setAge(Integer age) {this.age = age;}
}
我们再写一个测试类,来获取Student类中的相关信息,代码如下:
// 获取Student类的Class对象
Class clazz = Class.forName("com.soul.fanshe.Student");
// 获取Student类的所有方法信息
Method [] method = clazz.getDeclaredMethods();
System.out.println(clazz.getName() + "类的所有方法:");
for(Method m:method){
System.out.println(m.toString());
}
// 获取Student类的所有成员属性信息
Field[] field = clazz.getDeclaredFields();
System.out.println(clazz.getName() + "类的所有成员属性信息:");
for(Field f : field){
System.out.println(f.toString());
}
// 获取Student类的所有构造方法信息
Constructor [] constructor = clazz.getDeclaredConstructors();
System.out.println(clazz.getName() + "类的所有构造方法信息:");
for(Constructor c : constructor){
System.out.println(c.toString());
}
/**
* 输入结果如下
**/
com.soul.fanshe.Student类的所有方法:
public java.lang.String com.soul.fanshe.Student.getName()
public void com.soul.fanshe.Student.setName(java.lang.String)
public void com.soul.fanshe.Student.setAge(java.lang.Integer)
public void com.soul.fanshe.Student.setHight(java.lang.String)
public java.lang.String com.soul.fanshe.Student.getHight()
public java.lang.Integer com.soul.fanshe.Student.getAge()
com.soul.fanshe.Student类的所有成员属性信息:
private java.lang.String com.soul.fanshe.Student.name
private java.lang.String com.soul.fanshe.Student.hight
private java.lang.Integer com.soul.fanshe.Student.age
com.soul.fanshe.Student类的所有构造方法信息:
public com.soul.fanshe.Student()
public com.soul.fanshe.Student(java.lang.String,java.lang.String,java.lang.Integer)
4. 创建对象的两种方法
4.1. Class对象的newInstance()
使用Class对象的newInstance()方法来创建该Class对象对应类的实例,但是这种方法要求该Class对象对应的类有默认的空构造器。
Class sClazz = Class.forName("com.soul.fanshe.Student");
// 使用newInstance方法创建对象
// 如果创建实例的类无空构造器,则会抛出异常
Student s1 = (Student) sClazz.newInstance();
4.2. Constructor对象newInstance()方法
先使用Class对象获取指定的Constructor对象,再调用Constructor对象的newInstance()方法来创建Class对象对应类的实例,通过这种方法可以选定构造方法创建实例。
// 获取构造方法创建对象
Constructor c = sClazz.getDeclaredConstructor(String.class, String.class, Integer.class);
// 创建对象并设置属性
Student s2 = (Student) c.newInstance("张三","185", 20);