1.在Java中,反射是通过包 java.lang.reflect.* 来实现的。
2.对反射的理解:Reflection被视为动态语言的关键,反射机制允许程序在执行期间借助Reflection API取得任何类的内部信息,并能直接操作任意对象的内部属性及方法
首先定义一个Person类
public class Person {
public String name;
public int age;
public String sex;
private int fd;
public Person() {
System.out.println(
"dfdfa"
);
}
public Person(String name, int age,String sex) {
System.out.println("fdfd");
this.name = name;
this.age = age;
this.sex = sex;
}
@Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
", age=" + age +
", sex='" + sex + '\'' +
", fd=" + fd +
'}';
}
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;
}
private String show(String nation) {
System.out.println("我是一个"+nation+"人");
return nation;
}
private static void showDesc() {
System.out.println("我爱北京天安门");
}
}
3.获取class实例的几种方式:(Person代表一个类名)
1.Class<类名> clazz1 = 类名.class;
2.Person p1 = new Person();
Class<? extends Person> clazz2 = p1.getClass();
3.Class<?> clazz3 = Class.forName("Person");
4.ClassLoader classLoader = Reflact.class.getClassLoader();
Class<?> clazz4 = classLoader.loadClass("Person");
4.获取运行时类的完整结构:
1)获取(操作)某个类(Person)的指定属性:
Class clazz = Person.class;
Person o = (Person) clazz.newInstance();//生成一个当前类(Person)的对象实例
Field name = clazz.getDeclaredField("name");//获取class实例中的所有字段,不包括所继承的类中的字段
name.setAccessible(true);//让运行时类中的非public方法可以进行修改
name.set(o,"王虎");
System.out.println(name.get(o));
2)获取(操作)某个类(Person)的方法(非静态方法,需要创建类的对象实例):
Class clazz = Person.class;
//创建运行时类的对象
Person o = (Person) clazz.newInstance();
//获取指定的某个方法
//getDeclaredMethod() :参数一:指定获取的方法的名称 参数二:指定获取的方法的形参列表
Method show = clazz.getDeclaredMethod("show", String.class);
show.setAccessible(true);
//invoke() 参数一:方法调用者 参数二:给方法形参赋值的实参
//invoke()方法返回的值即为方法返回值的类型
String returns = (String) show.invoke(o, "中国");
3)获取(操作)某个类(Person)的方法(静态方法,不需要创建类的对象实例):
Class clazz = Person.class;
Method showDesc = clazz.getDeclaredMethod("showDesc");//该静态方法是无参的
showDesc.setAccessible(true);
//由于此时showDesc()就是clazz类中的方法(静态方法时属于类的,而非静态方法属于对象,需要知道那个对象),
// 因此方法调用者可以写null或者Person.class,并且invoke()返回值为null
Object invoke = showDesc.invoke(null);
4)获取(操作)某个类(Person)的构造器(有参数的):
Class clazz = Class.forName("Person");
Person o =(Person) clazz.getConstructor(String.class,int.class,String.class).newInstance("A",1,"B");
其他知识了解
1.反射机制和面向对象中的封装性是不是矛盾的?
答:不矛盾。封装性提供了公有方法和私有方法,只是建议去使用公有方法。反射机制可以调用私有方法,但是一般调用公有方法更好
2.类的加载过程:程序经过javac.exe命令之后,生成一个或多个字节码文件(.class文件),接着java.exe对某个字节码文件进行解释运行,相当于某个字节码文件加载到内存中。此过程就称为类的加载。加载到内存中的类,就称为运行时类,此运行时类,
就作为Class的一个实例
3.加载到内存中的运行时类,会缓存一定的时间。在此时间之内,我们可以通过不同的方式来获取此运行时类
4.对Class的理解:
Class实例对应着加载到内存中的一个运行时类
创建运行时类对象时应该注意的几个点:
1.运行时类中必须要有空参构造器
2.该空参构造器一般是public,可以供外部反射器创造对象,不可以是private
* javaBean一般都要有一个空参构造器
* 一是让外部反射器调用,二是让它的子类进行super()时可以调用空参构造器