- 静态编译:在编译时确定类型,绑定对象即通过。
- 动态编译:运行时确定类型,绑定对象。动态编译最大限度地发挥了Java的灵活性,体现了多态的应用,可以减低类之间的耦合性。
使用反射可以赋予 JVM 动态编译的能力,否则类的元数据信息只能用静态编译的方式实现。
二. 反射API
在Java中,JDK为我们提供了一些反射相关的API,如下表所示:
类 | 含义 |
java.lang.Class | 代表整个字节码。代表一个类型,代表整个类。 |
java.lang.reflect.Method | 代表字节码中的方法字节码。代表类中的方法。 |
java.lang.reflect.Constructor | 代表字节码中的构造方法字节码。代表类中的构造方法。 |
java.lang.reflect.Field | 代表字节码中的属性字节码。代表类中的成员变量(静态变量+实例变量)。 |
接下来我们就来看看反射到底怎么用。
三. 具体使用
下面壹哥就用一个案例来让大家感受一下反射的牛逼之处。
1. 常规实现
我们知道,在Java中的实体类总会有一些固定的方法,比如每个属性的 get()、set()方法,还有初始化属性创建对象的构造方法,打印对象信息的toString()等方法。假如我们在没有使用注解的情况下,需要创建2个普通的实体类:Cat、Dog,代码如下所示:
public class Cat {
private String name;
private int age;
public Cat() {
}
public Cat(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
@Override
public String toString() {
return "Cat{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
}
public class Dog{
private String name;
private int age;
public Dog() {
}
public Dog(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
@Override
public String toString() {
return "Dog{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
}
我们想在案例代码中创建对象并打印对象信息****,如下所示:
public class Demo01 {
public static void main(String[] args){
Cat cat = new Cat("招财", 1);
System.out.println(cat.toString());
Dog dog = new Dog("旺财",2);
System.out.println(dog.toString());
}
}
在上面这两个实体类中,都存在着名字相同、但方法体不同的toString()方法。如果我们在实体类中不重写toString() ,直接通过对象调用 toString(),打印的结果不会是对象的信息。
我们可以使用反射给两个实体类,甚至更多的实体类自动加上toString()方法,从而达到减少代码量的目的。
2. 反射实现
2.1 创建父类BaseEntity
首先我们创建一个父类BaseEntity
public class BaseEntity {
@Override
public String toString() {
//1.获取反射对象
Class<? extends BaseEntity> clazz = this.getClass();
//2.创建 StringBuffer 对象拼接字符串
StringBuffer sb = new StringBuffer();
//3.通过 getSimpleName() 获取类名并拼接
sb.append(clazz.getSimpleName());
//拼接大括号
sb.append("{");
//4.获取所有成员变量对象
Field[] fields = clazz.getDeclaredFields();
Object value = null;
for (int i = 0; i < fields.length; i++) {
//5.获取成员变量对象
Field field = fields[i];
//6.打开访问权限
field.setAccessible(true);
//7.通过 getName() 获取属性名并拼接
sb.append(getName());
sb.append("=");
//8.通过 get() 传递 this 获取对象的属性值
try {
value = field.get(this);
} catch (IllegalAccessException e) {
e.printStackTrace();
}
//9.判断是否为 String 类型的属性,是则添加单引号
if(field.getType() == String.class){
sb.append("'");
sb.append(value);