反射(Reflection)能够让运行于JVM中的程序检测和修改运行时的行为。
自省是程序检测的行为。自省应该说是属于反射的一部分。
反射非常强大,它甚至能直接操作程序的私有属性。我们前面学习都有一个概念,private的只能类内部访问,外部是不行的,但这个规定被反射赤裸裸的打破了。它可以在运行时获取一个类的所有信息,可以获取到任何定义的信息(包括成员变量,成员方法,构造器等),并且可以操纵类的字段、方法、构造器等部分。
反射对象的创建
三种方法
Class c1 = Student.class;
Class c2 = new Student().getClass();
Class c3 = Class.forName("day16.Student");
获取类的基本信息
//获取包名
System.out.println(c1.getPackage().getName());//包名
//获取类名
System.out.println(c1.getSimpleName());//类名
//获取完整类名
System.out.println(c1.getName());//完整类名
获取构造方法的基本属性
//获取所有的构造方法
Constructor[] onstructors = c1.getConstructors();
//获得公开的构造方法
getConstructor(参数类型列表)
获取普通方法的基本属性
//获得所有可见的方法,包括继承的方法
getMethods();
//获得特定的一个方法
getMethod(方法名,参数类型列表)
for(Method m : methods) {
System.out.println(m);
//获取方法名称
System.out.println(m.getName());
//获取返回类型
System.out.println(m.getReturnType());
//获取方法的参数列表类型
System.out.println(Arrays.toString(m.getParameterTypes()));
}
获取成员变量的基本属性
Field[] fields = c1.getFields();
for(Field f :fields) {
System.out.println(f);
//获取变量名
System.out.println(f.getName());
//获取变量类型
System.out.println(f.getType());
}
调用方法
注意在调用重载方法的时候要注意,getMethod方法的参数类型变量八大基本类型为(类型.class)其他类为(完整类名)
Class c = Student.class;
//利用反射new对象
Object o = c.newInstance();
//返回的是toString的方法返回值
System.out.println(o);
//获取方法
Method m = c.getMethod("show");
m.setAccessible(true);
m.invoke(o);
Method m1 = c.getMethod("show",int.class);
m.invoke(o,1);
调用变量
Class<Student> c = Student.class;
Object o = c.newInstance();
Field f = c.getField("name");
f.set(o, "wxn");//给属性赋值1.对象,2.值
System.out.println(f.get(o));//得到name的值
暴力反射
其实在上述的反射过程中,我们会发现,上述的反射只能用于公共方法变量(public修饰的),而实际上我们还有一种方式是是去获得所有的变量与方法(包括私有的)
方法就是在每一个普通方法的名称中加入Declared举个例子:
getMethods()//是获取所有公共方法的方法
getDeclaredMethods()//是获取所有方法的方法
我们来举一个例子:
public class Person {
private String name;
private int age;
private void show() {
System.out.println(age);
}
}
Class<Person> c2 = Person.class;
Object o1 = c2.newInstance();
Field f = c2.getDeclaredField("age");
f.setAccessible(true);
f.set(o1, 18);;
System.out.println(f.get(o1));
}
注意代码f.setAccessible(true);
为了防止我们动用私有变量/方法,加入了这个访问权限,只有访问权限是true才可以访问