说到反射,首先要说一下Java中的类和对象。

在Java中万事万物皆对象(有两个 例外,一个是普通数据类型,另一个是静态的东西,静态的东西不是对象的,是属于类的)。

在Java中,类也是对象,类是java.lang.class类的实例对象,即所谓There is a class named Class。

 

以下代码说明了Java中Class类的实例对象的三种表达方式



package org.guyezhai.reflect;

public class ClassDemo1 {
    
    public static void main(String[] args) {
        // Foo的对象如何表示
        Foo foo1 = new Foo();
        // Foo这个类 也是一个实例对象,Class类的实例对象
        // 任何一个类都是Class类的实例对象,这个实例对象有三种表示方式
        
        // 第一种表示方式,任何一个类都有一个隐含的静态成员变量class
        Class c1 = Foo.class;
        
        // 第二种表达方式,已经知道该类的对象,通过getClass()方法
        Class c2 = foo1.getClass();
        
        // 官网中,c1、c2表示了Foo类的类类型(class type)
        // 万事万物皆对象,类也是对象是Class类的实例对象,称为该类的类类型
        
        // c1和c2都代表了Foo类的类类型,一个类只可能是Class类的一个实例对象
        System.out.println(c1 == c2);//true
        
        // 第三种表达方式
        Class c3 = null;
        try {
            c3 = Class.forName("org.guyezhai.reflect.Foo");
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }
        System.out.println(c2 == c3);//true
        
        // 我们完全可以通过类的类类型,创建该类的对象实例:通过c1、c2、c3创建Foo的实例
        try {
            Foo foo = (Foo) c1.newInstance();// 需要有无参构造函数
            foo.print();
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }
    }
}

class Foo {
    void print() {
        System.err.println("foo");
    }
}



 

java中的类加载分为静态加载类和动态加载类两种,先来一个静态加载类的例子:



package org.guyezhai.reflect;

public class Office {
    
    /**
     * @param args
     */
    public static void main(String[] args) {
        if ("Word".equals(args[0])) {
            // new对象是静态加载类,在编译时刻就需要加载所有的可能使用到的类
            // 通过动态加载类可以解决该问题
            Word w = new Word();
            w.start();
        }
        if ("Excel".equals(args[0])) {
            // Excel类不存在,编译不通过
            // Excel e = new Excel();
            // e.start();
        }
    }
    
}



这种加载方式是我们常用的,但是有一个问题,如果Word类或Excel类有一个不存在,就会编译不通过,导致整个程序无法运行。

在某些情况下这也许不是我们想要的,我们希望只要Word或Excel中任一个存在就可以使用,这就需要改为动态加载类方式,代码如下:

首先创建一个接口officeAble:



package org.guyezhai.reflect;

public interface OfficeAble {
    public void start();
}



然后让Word和Excel类实现OfficeAble接口:



package org.guyezhai.reflect;

public class Word implements OfficeAble {
    
    public void start() {
        System.out.println("word...start()");
    }
    
}



package org.guyezhai.reflect;

public class Excel implements OfficeAble {
    
    public void start() {
        System.out.println("excel...start()");
    }
    
}



这样就可以动态加载Word或Excel了:



package org.guyezhai.reflect;

public class officeBetter {
    
    /**
     * @param args
     */
    public static void main(String[] args) {
        try {
            // 动态加载类,在运行时刻加载
            Class c = Class.forName(args[0]);
            // 通过类类型,创建该类的对象
            OfficeAble oa = (OfficeAble) c.newInstance();
            oa.start();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
    
}