反射(reflect)机制是一种动态执行方法,访问及设置属性等的方法,在以后Spring,Hibernate等的学习当中会更深层次的学习这种机制;
反射就是把Java类中的各种成分映射成相应的java类,一个Java类中用一个类来表示,一个类中的组成部分:成员变量,方法,构造方法,修饰符,包等等信息也用一个个的Java类来表示,就像汽车是一个类,汽车中的发动机,变速箱等等也是一个个的类。表示java类的Class类显然要提供一系列的方法,来获得其中的变量,方法,构造方法,修饰符,包等信息,这些信息就是用相应的类来表示的,它们是Field、Method、Constructor、Package等。
下面简单介绍Class提供的一系列对类的Field、Method、Constructor、Package等进行操作的方法。
(1) 生成一个新的实例;String str = Class.forName("java.lang.String").newInstance();这个方法相当调用了String类默认的Constructor;
Constructor constructor = Class.forName("java.lang.String").getConstructor(StringBuffer.class);
String str = (String) constructor.newInstance("miracle");
相当于调用了String(StringBuffer)这个构造函数;
(2) 调用一个类的Method
Method method = Class.forName("ClassNameStr").getMethod("charAt", int.class);
method.invoke(str, 2);
如果是静态方法,则调用该方法的时候在method.invoke(null,args);第一个Obj参数为Null;
(3) 访问一个类的Field(公共和私有)
Id 是Field类的公共属性,birth是Field类的私有属性;
FieldTest ft = new FieldTest();
Field id = Class.forName("FieldTest").getField("id");
id.get(ft);
Field birth = Class.forName("FieldTest").getDeclaredField("birthday");
birth.setAccessible(true);
birth.get(ft);
(4) 内省(一个JavaBean就是其中的方法符合某种命名规则的一个java类,
JavaBean的属性是根据其中的setter和getter方法来确定的,而不是根据其中的成员变量。在Java EE开发中,经常要使用到JavaBean,JDK中提供了对JavaBean进行操作的一些API。
我setId,中文意思即设置id,至于你把它存到哪个变量上,我管吗?我getId,中文
意思即取id,至于你从哪个变量上取,我管吗?所以,一个类被当作javaBean使用时,
JavaBean的属性是根据方法名推断出来的,它根本看不到java类内部的成员变量。去掉set前缀,然后取剩余部分,如果剩余部分的第二个字母是小写的,则把剩余部分的首字母改成小的。 setId的属性名是id,setid的属性名是id,setCPU的属性名是什么,CPU,setUPS的属性名是什么?UPS)
User user = new User();
BeanInfo beaninfo = Introspector.getBeanInfo(User.class);
PropertyDescriptor[] propertyDescriptors = beaninfo.getPropertyDescriptors();
for(PropertyDescriptor propertyDescriptor:propertyDescriptors){
if(propertyDescriptor.getName().equals("id")){
Method setId = propertyDescriptor.getWriteMethod();
setId.invoke(user, 5);
}
}
(5) 写一个程序,这个程序能够根据用户提供的类名,去执行该类中的main方法。
String className = args[0];
Method mainMethod = Class.forName(className).getMethod("main", String[].class);
mainMethod.invoke(null, new Object[]{new String[]{"main"}});
注意此时对于提供类名的main函数的String[] args 参数的处理,在给main方法传递参数时,不能使用如下代码mainMethod.invoke(null,new String[]{"xxx"}),虽然这符合jdk1.5的语言要求,但是jdk把它理解成了jdk1.4的样子,所以,出现了参数类型不对的问题。对于传数组的情况,只能按jdk1.4的语法进行调用,因为如果参数类型就是Object,它是没有办法区别的;如果它是String args,则
Method mainMethod = Class.forName(className).getMethod("main", String.class);
mainMethod.invoke(null,new String(“main”));
如果此时提供的类的main函数参数为String args1,String args2,则其处理方式为:
Method mainMethod = Class.forName(className).getMethod("main", String.class,String.class);
mainMethod.invoke(null,new String(“main”),new String(“mainagain”));