简单总结Java的反射机制的运用
作者:turkeyzhou
Java 的反射机制是使其具有动态特性的非常关键的一种机制,也是在JavaBean 中广泛应用的一种特性。
简单来说,一个类或者一个对象是拥有下面几种属性的:
Method,Constructor,Field,其大致结构类图如下:
我们现在用代码来说明问题:
首先,我们看Class类,在Class类中,我们可以看见下面的几个重要的方法;
- getInterfaces()
- getSuperClass();
- isInterface();
这是用来得到一个类的接口或者超类,以及判断这个类是不是一个接口;
- forName(String className);根据一个类名得到一个相应的类类型;
- getClassLoader();得到这个类相应的类加载器;
以及下面的几种方法:
getField(Method/Constructor)(s)(name); getDeclaredField(Method/Constructor)(s)();
分别是获取这个类的相应的Constructor,field,method的;
区别在于,含有Declared的方法能够得到这个类所声明的所有的属性,而没有Declared的只能得到公共public的属性;
而继承了Member分别赋予了这个三个类能够得到声明其的Class,用getDeclaringClass();在这里我们再次介绍一下Modifiers,我们都知道在field或者Constructor,Method前面都含有若干修饰符,如:
public static final String name="corey";
等等,我们应用getModifiers()能够拿到这个修饰符的一个整形值,然后应用Modifier这个类的静态方法来进行判断;如:
Modifier.isStatic(int)等等;
接下来,我们来看看AccessibleObject的几个主要的方法,AccessibleObject中主要的几个方式第一是
getAnnotation();得到某个属性的注释;
isAccessible();能否访问;如果不能访问,我们可以采取setAccessible(boolean)来设置其的可访问性;(这个我们在spring中看到过);
然后我们分别来看看这个三个类一些重要的特性:
Constructor:
- newInstance(args):能够使用这个构造器得到一个类的实例;
Field:
- getType();得到这个字段的类;
- set/get(Object,value):一系列的基本类型字段的设置方法或者Object的设置方法;
Method:
- getParameterTypes();得到所有参数的类型;
- getExceptionTypes();得到所有抛出异常的类型;
- invoke(Object,args);调用Object对象的这个方法,参数是args;
下面是一份实例代码:
package org.corey.demo; public interface IName { public String getFirstName(); public void setFirstName(String firstName); public String getLastName(); public void setLastName(String lastName); } package org.corey.demo; public class Name { private String firstName; private String lastName; public String publicName; public Name(String firstName, String lastName) { this.firstName = firstName; this.lastName = lastName; } public Name() { } public String getFirstName() { return firstName; } public void setFirstName(String firstName) { this.firstName = firstName; } public String getLastName() { return lastName; } public void setLastName(String lastName) { this.lastName = lastName; } } package org.corey.demo; import java.lang.reflect.Constructor; import java.lang.reflect.Field; import java.lang.reflect.Method; public class Demo { /** * @param args */ public static void main(String[] args) { try { Class clazz = Class.forName("org.corey.demo.Name"); Constructor con = clazz.getDeclaredConstructor(new Class[] { String.class, String.class }); Name corey = (Name)(con.newInstance("corey", "zhou")); System.out.println(corey.getFirstName()+" "+corey.getLastName()); Field[] fields=clazz.getDeclaredFields(); for(int index=0;index<fields.length;index++){ System.out.println(fields[index].getName()+" accessible "+fields[index].isAccessible()); } Method[] methods=clazz.getDeclaredMethods(); for(int index=0;index<methods.length;index++){ System.out.println(methods[index].getName()); } Field field=clazz.getDeclaredField("firstName"); if(!field.isAccessible()){ field.setAccessible(true); field.set(corey, "syna"); } Method method=clazz.getDeclaredMethod("setLastName", new Class[]{String.class}); method.invoke(corey, "wang"); System.out.println(corey.getFirstName()+" "+corey.getLastName()); } catch (Exception e) { // TODO Auto-generated catch block e.printStackTrace(); } } }
console:
corey zhou firstName accessible false lastName accessible false publicName accessible false getFirstName getLastName setLastName setFirstName syna wang