1 反射的应用场合
在编译时根本无法知道该对象或类可能属于哪些类,程序只依靠运行时信息来发现该对象和类的真实信息
2 反射的作用
通过反射可以使程序代码访问装载到JVM 中的类的内部信息
获取已装载类的属性信息
获取已装载类的方法
获取已装载类的构造方法信息
3在JDK中,主要由以下类来实现Java反射机制,这些类都位于java.lang.reflect包中
Class类:代表一个类
Field 类:代表类的成员变量(属性)
Method类:代表类的成员方法
Constructor 类:代表类的构造方法
Array类:提供了动态创建数组,以及访问数组的元素的静态方法
4 Class类是Java 反射机制的起源和入口
用于获取与类相关的各种信息
提供了获取类信息的相关方法
Class类继承自Object类
Class类是所有类的共同的图纸
每个类有自己的对象,好比图纸和实物的关系
每个类也可看做是一个对象,有共同的图纸Class,存放类的结构信息,能够通过相应方法取出相应信息
类的名字
属性
方法
构造方法
父类和接口
文件名 说 明
getFields() 获得类的public类型的属性。
getDeclaredFields() 获得类的所有属性
getField(String name) 获得类的指定属性
getMethods() 获得类的public类型的方法
getMethod (String name,Class [] args) 获得类的指定方法
getConstrutors() 获得类的public类型的构造方法
getConstrutor(Class[] args) 获得类的特定构造方法
newInstance() 通过类的无参构造方法创建该类的一个对象
getName() 获得类的完整名字
getPackage() 获取此类所属的包
getSuperclass() 获得此类的父类对应的Class对象
5为一个class生成对应的Class对象
方 法 示 例
对象名.getClass() String str="bdqn";
Class clazz = str.getClass();
对象名.getSuperClass() Student stu = new Student();
Class c1 = stu.getClass();
Class c2 = stu.getSuperClass();
Class.forName() Class clazz = Class.forName("java.lang.Object");
Class.forName("oracle.jdbc.driver.OracleDriver");
类名.class Class c1 = String.class;
Class c2 = Student.class;
Class c2 = int.class
包装类.TYPE Class c1 = Integer.TYPE;
Class c2 = Boolean.TYPE;
6使用反射动态创建对象
方法1:通过Class的newInstance()方法
该方法要求该Class对象的对应类有无参构造方法
执行newInstance()实际上就是执行无参构造方法来创建该类的实例
public class ConstructorTest {
public static void main(String[] args) throws Exception {
//方法1:调用Class的newInstance方法,仅适用于无参构造方法
Class clazz=Class.forName("cn.jbit.reflection.Student");
Object obj=clazz.newInstance();
}
}
方法2:通过Constructor的newInstance()方法
先使用Class对象获取指定的Constructor对象
Class clazz3=Student.class;
Constructor c=clazz3.getConstructor(String.class,int.class);
Object stu=c.newInstance("张三",17);
7使用反射动态修改查询属性值
通过Class对象的getFields()或者getField()方法可以获得该类所包括的全部Field属性或指定Field属性。Field类提供了以下方法来访问属性
getXxx(Object obj):获取obj对象该Field的属性值。此处的Xxx对应8个基本数据类型,如果该属性类型是引用类型则直接使用get(Object obj)
setXxx(Object obj,Xxx val):将obj对象的该Field赋值val。此处的Xxx对应8个基本数据类型,如果该属性类型是引用类型则直接使用set(Object obj, Object val)
setAccessible(Boolean flag):若flag为true,则取消属性的访问权限控制,即使private属性也可以进行访问
例:
public class FieldTest {
public static void main(String[] args) throws Exception {
Class clazz = Class.forName("cn.jbit.reflection.Student");
Object obj = clazz.newInstance();
// 调用getDeclaredField("name") 取得name属性对应的Field对象
Field f = clazz.getDeclaredField("name");
// 取消属性的访问权限控制,即使private属性也可以进行访问。
f.setAccessible(true);
// 调用get()方法取得对应属性值。
System.out.println(f.get(obj)); //相当于obj.getName();
// 调用set()方法给对应属性赋值。
f.set(obj, "lkl"); //相当于obj.setName("lkl");
// 调用get()方法取得对应属性修改后的值。
System.out.println(f.get(obj));
}
}
8 使用反射动态执行方法
通过Class对象的getMethods() 方法可以获得该类所包括的全部方法, 返回值是Method[]
通过Class对象的getMethod()方法可以获得该类所包括的指定方法, 返回值是Method
每个Method对象对应一个方法,获得Method对象后,可以调用其invoke() 来调用对应方法
Object invoke(Object obj,Object [] args):obj代表当前方法所属的对象的名字,args代表当前方法的参数列表,返回值Object是当前方法的返回值,即执行当前方法的结果。
例子:
public class TestMethod {
public int add(int x, int y) { return x + y; }
public void shout(String name) {System.out.println("my name is"+name);}
public static void main(String[] args) throws Exception {
// 创建该类的一个对象
Class clazz = TestMethod.class;
Object obj = clazz.newInstance();
// 调用该对象的add方法
Method amethod=clazz.getMethod("add",new Class[]{int.class, int.class});
Object result = amethod.invoke(obj, new Object[] { 5, 7 });
System.out.println(result);
// 调用该对象的shout方法
Method smethod=clazz.getMethod("shout",new Class[]{String.class});
smethod.invoke(obj, new Object[] { "lkl" });
}
}
反射技术优缺点
反射提高了Java程序的灵活性和扩展性,降低耦合性,提高自适应能力。它允许程序创建和控制任何类的对象,无需提前硬编码目标类
反射是其它一些常用语言,如C、C++、Fortran 或者Pascal等都不具备的
Java反射技术应用领域很广,如软件测试、 EJB、JavaBean等
许多流行的开源框架例如Struts、Hibernate、Spring在实现过程中都采用了该技术
反射的缺点
性能问题
使用反射基本上是一种解释操作,用于字段和方法接入时要远慢于直接代码。因此Java反射机制主要应用在对灵活性和扩展性要求很高的系统框架上,普通程序不建议使用。
使用反射会模糊程序内部逻辑
程序人员希望在源代码中看到程序的逻辑,反射等绕过了源代码的技术,因而会带来维护问题。反射代码比相应的直接代码更复杂。 |
|