反射(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才可以访问