一:反射机制
Java反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为Java语言的反射机制。
二:反射机制的作用
1 反编译:.class-->.java
2 通过反射机制访问java对象的属性,方法,构造方法等;
三:sun为我们提供了那些反射机制中的类:
java.lang.Class;
java.lang.reflect.Constructor; java.lang.reflect.Field;
java.lang.reflect.Method;
java.lang.reflect.Modifier;
四:具体实现
1 反射机制获取类有三种方法,我们来获取Employee类型
1 //第一种方式: 2 Class c1 = Class.forName("Employee"); 3 //第二种方式: 4 //java中每个类型都有class 属性. 5 Classc2 = Employee.class; 6 //第三种方式: 7 //java语言中任何一个java对象都有getClass 方法 8 Employee e = new Employee(); 9 Classc3 = e.getClass(); //c3是运行时类 (e的运行时类是Employee)
2 创建对象:获取类以后我们来创建它的对象,利用newInstance:
1 Class c =Class.forName("Employee"); 2 //创建此Class 对象所表示的类的一个新实例 3 Objecto = c.newInstance(); //调用了Employee的无参数构造方法.
3 获取属性:分为所有的属性和指定的属性:
a,先看获取所有的属性的写法:
//获取整个类 Class c = Class.forName("java.lang.Integer"); //获取所有的属性 Field[] fs = c.getDeclaredFields(); //定义可变长的字符串,用来存储属性 StringBuffer sb = new StringBuffer(); //通过追加的方法,将每个属性拼接到此字符串中 //最外边的public定义 sb.append(Modifier.toString(c.getModifiers()) + " class " + c.getSimpleName() +"{ ");//获取修饰符public class //里边的每一个属性 for(Field field:fs){ sb.append(" ");//空格
sb.append(Modifier.toString(field.getModifiers())+" ");//获得属性的修饰符,例如public,static等等 sb.append(field.getType().getSimpleName() + " ");//属性的类型的名字 sb.append(field.getName()+"; ");//属性的名字+回车 } sb.append("}"); System.out.println(sb);
b,获取特定的属性:
1 public static void main(String[] args) throws Exception{ 2 <span style="white-space:pre"> </span>//以前的方式: 3 /* 4 User u = new User(); 5 u.age = 12; //set 6 System.out.println(u.age); //get 7 */ 8 //获取类 9 Class c = Class.forName("User"); 10 //获取id属性 11 Field idF = c.getDeclaredField("id"); 12 //实例化这个类赋给o 13 Object o = c.newInstance(); 14 //打破封装 15 idF.setAccessible(true); //使用反射机制可以打破封装性,导致了java对象的属性不安全。 16 //给o对象的id属性赋值"110" 17 idF.set(o, "110"); //set 18 //get 19 System.out.println(idF.get(o)); 20 }
4,获取方法,和构造方法,不再详细描述,只来看一下关键字:
【例题】:下列代码的执行结果:
1 package test; 2 import java.util.Date; 3 public class SuperTest extends Date{ 4 private static final long serialVersionUID = 1L; 5 private void test(){ 6 System.out.println(super.getClass().getName()); 7 } 8 public static void main(String[]args){ 9 new SuperTest().test(); 10 } 11 }
TestSuper和Date的getClass()都没有重写,他们都是调用Object的getClass(),而Object的getClass()作用是返回的是运行时的类的名字。
这个运行时的类就是当前类。super.getClass().getName() 返回的是test.SuperTest。在调用getName()方法而getName()返回:包名+类名。
普通的java对象是通过new关键字把对应类的字节码文件加载到内存,然后创建该对象的。
反射是通过一个名为Class的特殊类,用Class.forName("className");得到类的字节码对象,然后用newInstance()方法在虚拟机内部构造这个对象(针对无参构造函数)。
也就是说反射机制让我们可以先拿到java类对应的字节码对象,然后动态的进行任何可能的操作,
包括
- 在运行时判断任意一个对象所属的类
- 在运行时构造任意一个类的对象
- 在运行时判断任意一个类所具有的成员变量和方法
- 在运行时调用任意一个对象的方法
使用反射的主要作用是方便程序的扩展。