java反射

来源:互联网 发布:淘宝卖家卖出多久到账 编辑:程序博客网 时间:2024/06/02 02:53

反射

      反射就是把Java类中的各种成分映射成相应的java类。例如,一个Java类中用一个Class类的对象来表示,一个类中的组成部分:成员变量,方法,构造方法,包等等信息也用一个个的Java类来表示,就像汽车是一个类,汽车中的发动机,变速箱等等也是一个个的类。表示Java类的Class类显示要提供一系列的方法,来获得其中的变量,方法,构造方法,修饰符,包等信息,这些信息就是用相应类的实例对象来表示,它们是Field、Method、Contructor、Package等等。

      一个类中的每个成员都可以用相应的反射API类的一个实例对象来表示,通过调用Class类的方法可以的得到这些实例对象后,得到这些实例对象后有什么用呢?怎么用呢?这正是学习和应用反射的要点。

构造方法的反射应用

Consturctor(构造器)类代表某个类中的一个构造方法

@1@ 得到某个类所有的构造方法:例如:Constructor [] constructors = Class.forName("java.lang.String").getConstructors();

@2@ 得到某一个构造方法:例如:Constructor constructor = Class.forName("java.lang.String").getConstructor(StringBuffer.class);

@3@ 创建实例对象:通常方式:String str=new String(new StringBuffer("abc"));
                            反射方式:String str=(String) constructor.newInstance(new StringBuffer("abc"));

Java反射机制主要提供了以下功能: 在运行时判断任意一个对象所属的类;在运行时构造任意一个类的对象;在运行时判断任意一个类所具有的成员变量和方法;在运行时调用任意一个对象的方法;生成动态代理。
1. 得到某个对象的属性

/** * 得到某个对象的属性 * @param owner * @param fieldName * @return * @throws Exception */@SuppressWarnings("rawtypes")public Object getProperty(Object owner, String fieldName) throws Exception {Class ownerClass = owner.getClass();Field field = ownerClass.getField(fieldName);Object property = field.get(owner);return property;}
Class ownerClass owner.getClass():得到该对象的Class。
Field field ownerClass.getField(fieldName):通过Class得到类声明的属性。
Object property field.get(owner):通过对象得到该属性的实例,如果这个属性是非公有的,这里会报IllegalAccessException。

2. 得到某个类的静态属性

Java代码
/** * 得到某个类的静态属性 * @param className * @param fieldName * @return * @throws Exception */@SuppressWarnings("rawtypes")public Object getStaticProperty(String className, String fieldName)throws Exception {Class ownerClass = Class.forName(className);Field field = ownerClass.getField(fieldName);Object property = field.get(ownerClass);return property;}
Class ownerClass Class.forName(className) :首先得到这个类的Class。

Field field ownerClass.getField(fieldName):和上面一样,通过Class得到类声明的属性。

Object property field.get(ownerClass) :这里和上面有些不同,因为该属性是静态的,所以直接从类的Class里取。

3. 执行某对象的方法

Java代码
/** * 执行某对象的方法 * @param owner * @param methodName * @param args * @return * @throws Exception */@SuppressWarnings({ "rawtypes", "unchecked" })public Object invokeMethod(Object owner, String methodName, Object[] args)throws Exception {Class ownerClass = owner.getClass();Class[] argsClass = new Class[args.length];for (int i = 0, j = args.length; i < j; i++) {argsClass[i] = args[i].getClass();}Method method = ownerClass.getMethod(methodName, argsClass);return method.invoke(owner, args);}
Class owner_class owner.getClass() :首先还是必须得到这个对象的Class。

5~9行:配置参数的Class数组,作为寻找Method的条件。

Method method ownerClass.getMethod(methodName, argsClass):通过methodName和参数的argsClass(方法中的参数类型集合)数组得到要执行的Method。

method.invoke(owner, args):执行该Method.invoke方法的参数是执行这个方法的对象owner,和参数数组args,可以这么理解:owner对象中带有参数args的method方法。返回值是Object,也既是该方法的返回值。

4. 执行某个类的静态方法

Java代码
/** * 执行某个类的静态方法 * @param className * @param methodName * @param args * @return * @throws Exception */@SuppressWarnings({ "rawtypes", "unchecked" })public Object invokeStaticMethod(String className, String methodName,Object[] args) throws Exception {Class ownerClass = Class.forName(className);Class[] argsClass = new Class[args.length];for (int i = 0, j = args.length; i < j; i++) {argsClass[i] = args[i].getClass();}Method method = ownerClass.getMethod(methodName, argsClass);return method.invoke(null, args);}

基本的原理和实例3相同,不同点是最后一行,invoke的一个参数是null,因为这是静态方法,不需要借助实例运行。

5. 新建实例

Java代码
/** * 新建实例 * @param className * @param args * @return * @throws Exception */@SuppressWarnings({ "rawtypes", "unchecked" })public Object newInstance(String className, Object[] args) throws Exception {Class newoneClass = Class.forName(className);Class[] argsClass = new Class[args.length];for (int i = 0, j = args.length; i < j; i++) {argsClass[i] = args[i].getClass();}Constructor cons = newoneClass.getConstructor(argsClass);return cons.newInstance(args);}

这里说的方法是执行带参数的构造函数来新建实例的方法。如果不需要参数,可以直接使用newoneClass.newInstance()来实现。

Class newoneClass Class.forName(className):第一步,得到要构造的实例的Class。

第5~第9行:得到参数的Class数组。

Constructor cons newoneClass.getConstructor(argsClass):得到构造子。

cons.newInstance(args):新建实例。

6. 判断是否为某个类的实例

Java代码
/** * 判断是否为某个类的实例 * @param obj * @param cls * @return */@SuppressWarnings("rawtypes")public boolean isInstance(Object obj, Class cls) {return cls.isInstance(obj);}

7. 得到数组中的某个元素

Java代码
/** * 得到数组中的某个元素 * @param array * @param index * @return */public Object getByArray(Object array, int index) {return Array.get(array, index);}
0 0