Java反射技术(详细)
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()时可以调用空参构造器