在上一篇文章,已经讲到了反射的一些基本概念以及基本的使用,也介绍到了一些反射中常用的方法以及区别,今天我们再次进入反射的第二期,更进一步的了解一下反射。
我们在使用反射去创建一个类的实例的时候一般都是使用:
Class clazz = Test.class;
Constructor constructor = clazz.getDeclaredConstructor();
constructor.setAccessible(true);
Object obj = constructor.newInstance();
这种写法稍显繁杂,我们可以通过另一种便捷的方式去实现:
Class clazz = Test.class;
Test instance1 = clazz.newInstance();
这种方式虽然不直接使用构造器对象,但是这种方式是有缺陷的:
1、如果需要实例化的Test类无参构造方法使用private修饰则会出现java.lang.IllegalAccessException异常,因为没有权限访问。
2、如果我们没有无参构造方法,因为newInstance()是不需要传参的,在执行时newInstance()只会寻找公共的、无参的构造方法。
注意:在这里我们一定要区分newInstance这个方法,这里的newInstance方法是Class下的一个方法,与Constructor中的newInstance方法是不一样的。
Method对象的使用
如果我们需要获取某一个类中的所有Method对象,我们还是需要通过字节码对象去操作。比如我们现在需要和获取到Test2类中的所有Method对象。
public class Test2 {
private String name;
private Integer age;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Integer getAge() {
return age;
}
public void setAge(Integer age) {
this.age = age;
}
@Override
public String toString() {
return "Test2{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
private void test(){
System.out.println("I'm test method");
}
}
我们在Test2中提供了get、set方法,也重写了toString方法,并且也创建了一个私有的test方法,然后再通过使用getDeclaredMethods()方法获取到了Test2中所有的Method对象。
Class clazz = Test2.class;
Method[] declaredMethods = clazz.getDeclaredMethods();
for (Method declaredMethod : declaredMethods) {
System.out.println(declaredMethod);
}
但是如果使用getMethods()这个时候我们就不能获取到私有Method对象了,如果我们需要获取指定的方法则可以使用getMethod(name)或者getDeclaredMethod(name),至于他们两个的区别就不再赘述。
那么又有一个问题出现了,Java是有多态特性的,如果我们有某一个方法进行了方法重载,那么这个时候我们如果根据方法名称去获取不就有问题了吗?不用急,我们可以通过getMethod(name,paramClassType)或者getDeclaredMethod(name,paramClassType)获取到我们指定的重载方法。如:
public class Test{
private void test(){
System.out.println("I'm test method");
}
private void test(String name, Integer age){
System.out.println("I'm test method:"+name+age+"岁");
}
}
// 通过此方式便可以获取到我们指定的重载方法
Method test = clazz.getDeclaredMethod("test", String.class, Integer.class);
获取到指定的方法后我们要如何使用?
我们一般在使用一个方法的时候都是obj.method(param);但是如果我们需要用反射进行调用则需要,method.invoke(obj,param);
Class clazz = Test2.class;
Method test = clazz.getDeclaredMethod("test", String.class, Integer.class);
test.setAccessible(true);
test.invoke(clazz.newInstance(), "zhangsan", 12);
Field对象如何使用呢?
这里其实与Class、Method对象都是大同小异了,都是同样的玩法。
Class clazz = Test2.class;
Field field1 = clazz.getDeclaredField("name");
Field field2 = clazz.getDeclaredField("age");
Test2 test2 = clazz.newInstance();
field1.setAccessible(true);
field1.set(test2, "张三");
field2.setAccessible(true);
field2.set(test2, 12);
System.out.println(test2);
至此我们的反射也就已经整体结束了,后续将继续努力给大家带来一些易懂的Java知识。如果你有兴趣,可以关注我后续将带来更多精彩的文章。