黑马程序员技术交流社区

标题: 反射 [打印本页]

作者: 黑马张健    时间: 2012-5-26 18:49
标题: 反射
反射在框架方面的作用,以及优缺点的体现。
请高手从框架方面,说一下反射的应用和优缺点。希望全面一些,谢谢!
作者: 黑马11期李项京    时间: 2012-5-26 19:56

为什么要用反射机制?直接创建对象不就可以了吗,这就涉及到了动态与静态的概念,
    静态编译:在编译时确定类型,绑定对象,即通过。
    动态编译:运行时确定类型,绑定对象。动态编译最大限度发挥了java的灵活性,体现了多
    态的应用,有以降低类之间的藕合性。
    一句话,反射机制的优点就是可以实现动态创建对象和编译,体现出很大的灵活性,特别是在J2EE的开发中
    它的灵活性就表现的十分明显。比如,一个大型的软件,不可能一次就把把它设计的很完美,当这个程序编
    译后,发布了,当发现需要更新某些功能时,我们不可能要用户把以前的卸载,再重新安装新的版本,假如
    这样的话,这个软件肯定是没有多少人用的。采用静态的话,需要把整个程序重新编译一次才可以实现功能
    的更新,而采用反射机制的话,它就可以不用卸载,只需要在运行时才动态的创建和编译,就可以实现该功
    能。
       它的缺点是对性能有影响。使用反射基本上是一种解释操作,我们可以告诉JVM,我们希望做什么并且它
    满足我们的要求。这类操作总是慢于只直接执行相同的操作。
更详细的请看http://wenku.baidu.com/view/9a49e20d763231126edb1188.html?from=rec&pos=0&weight=2&lastweight=1&count=4
反射在java中属于比较高级的技术,我现在也不太懂,以后会有更多接触的,互勉!
作者: 陆建平    时间: 2012-5-26 20:33

Java中的反射机制
.

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反射机制主要应用在对灵活性和扩展性要求很高的系统框架上,普通程序不建议使用。
使用反射会模糊程序内部逻辑
程序人员希望在源代码中看到程序的逻辑,反射等绕过了源代码的技术,因而会带来维护问题。反射代码比相应的直接代码更复杂。

作者: 张文建    时间: 2012-5-26 22:26
陆建平 发表于 2012-5-26 20:33
Java中的反射机制
.

总结得真不错,下去我得好好拜读了,呵呵
作者: 魏明明    时间: 2012-6-1 23:47
陆兄 很强大
作者: xiaoxiu1984    时间: 2012-6-11 20:10
感谢师兄哈,总结的不错




欢迎光临 黑马程序员技术交流社区 (http://bbs.itheima.com/) 黑马程序员IT技术论坛 X3.2