1. 首先理解什么是反射?
反射(Reflection)能够让运行于 JVM 中的程序检测和修改运行时的行为。
2. 为什么需要反射?
反射能够让我们:
- 在运行时检测对象的类型;
- 动态构造某个类的对象;
- 检测类的属性和方法;
- 任意调用对象的方法;
- 修改构造函数、方法、属性的可见性等;
3.Class 类实例对象的方法
Class 实例对象的三种方法:
a. 通过类的隐含的静态成员 class;
b. 通过该类的 getClass() 方法;
c. 通过类的全称,用 Class.forName(“ 类名”) 获取。
以上三种方法可以参考下面的代码:
class ClassDemo0{ public static void main(String[] args) { Foo foo1=new Foo(); //a, 任何一个类都有一个隐含的静态成员 class Class c1=Foo.class; //b, 已知该类,通过 getClass 方法 Class c2=foo1.getClass(); //注 c1,c2 表示 Foo 类的类类型(class type) System.out.println(c1==c2); //true, 一个类只可能是 Class 类的一个实例对象 //c, 通过类的全称获取 Class c3=null; try{ c3=Class.forName("Foo"); }catch(ClassNotFoundException e){ e.printStackTrace(); } System.out.println(c2==c3);//true //通过类的类类型创建该类的对象实例,通过 c1,c2,c3 创建 Foo 的对象实例 try{ Foo foo=(Foo)c1.newInstance();//前提:该类要有无参构造函数 }catch(InstantiationException e){ e.printStackTrace(); }catch(IllegalAccessException e){ e.printStackTrace(); } } } class Foo{ void print(){ System.out.println("Foo"); } }
4. 动态加载类
首先我们要了解:为什么要动态加载?
当我们编写一个程序的时候,我们首先要编写代码,然后进行编译,编译通过之后才可以运行。
然后我们要清楚,当我们使用 new 的方式来创建一个对象的时候 [ new Book() ],Book 这个类就要被静态加载。
(编译时刻加载的类是静态加载类,运行时刻加载的类是动态加载类)
现在,我们假设有这么一段代码:
class Office{ public static void main(String[] args) { //new 创建对象 是静态加载类,在编译时刻就需要加载所有的可能使用到的类,(若没有建 Word 类),编译不通过 //若有建 Word 类,但 Excel 类没建,但编译还是不能通过 //希望用到时再加载,可以通过动态加载类实现 if("Word".equals(args[0])){ Word w=new Word(); w.start(); } if("Excel".equals(args[0])){ Excel e=new Excel(); e.start(); } } }
再假设 Word 这个类是存在的,那么 new Word() 这句话编译是可以通过的。
但是如果 Excel 这个类不存在,那么编译就会报错,编译不通过,程序就不能运行。
也就是,因为一个类的错误,影响了整体程序的运行。
那么我们要避免这样的情况,应该怎么做呢?
在这种情况之下,就可以使用动态代理。
class OfficeBetter{ public static void main(String[] args) { try{ //动态加载类,在运行时加载 Class c=Class.forName(args[0]); //通过类类型,创建该类对象 //Word w=(Word)c.newInstance(); //如果加载的是 Excel 呢? //Excel e=(Excel)c.newInstance(); //如果加载的是 Excel 呢? //后期增加功能就不用改动 OfficeBetter,实现在线升级,扩展性更强 OfficeAble oa=(OfficeAble)c.newInstance(); //所以做一个标准————功能型的类尽量使用动态加载 oa.start(); }catch(Exception e){ e.printStackTrace(); } } } interface OfficeAble{ public void start(); } class Word implements OfficeAble{ public void start(){ System.out.println("word...start..."); } } class Excel implements OfficeAble{ public void start(){ System.out.println("Excel...start..."); } }
可以看到,正常情况下,如果需要在 officeBetter 这个类中通过 new 的方式将我们可能用到的所有的类都实例化出来。
这样不仅占用了内存,同时也存在如果一个出现错误其他的也不能运行这样的问题。
而我们现在通过一个借口,让子类去继承,然后我们通过获得传递过来的 class,创建出该类的对象,这个对象就是子类的对象了。
在这种情况下,如果有新增或者减少功能(比如在旧的方法的情况下要添加一个 pencil),就不需要对这个类进行修改,只需要新建新的类就可以了。
转载请注明:热爱改变生活.cn » 反射的介绍以及动态加载类——反射_1
本博客只要没有注明“转”,那么均为原创。 转载请注明链接:sumile.cn » 反射的介绍以及动态加载类——反射_1