java中的内省及javabean操作
在java的反射中,所有的类被抽象出一个类,即Class类,这样我们就可以在程序运行的过程中通过配置文件,动态的加载类。但是在用反射的时候有些前提,就是:当我们调用有参的constructor的时候必须先知道构造函数传入的参数是什么类型;调用Method的时候必须先知道成员函数传入的参数是什么类型;调用field的时候必须先知道成员变量的类型。但是在某些情况下,一个类中的成员属性的名字对外是不可见的,这时候我们只可以得出他的成员类型的数组field[],在不知道确切变量名的情况下不可以定位到确切的变量上。而他提供了对这个属性公开的读(get)、写(set)方法, 虽然方法名和方法操作的对象有时候有很大的关系,但是我们不能保证这个关系一定成立。而且这种只提供get、set方法,不提供变量名的情况还很常见,因此java就将这种抽象为javabean类,对javabean类的操作,通过get、set函数后面的名字就“拟”得出变量的名字。通过这个名字,我们可以确切的求出某个变量的值。而对javabean的操作是通过内省(introspector)来完成的。上例子:
1 package reflect;
2
3 import java.beans.BeanInfo;
4 import java.beans.Introspector;
5 import java.beans.MethodDescriptor;
6 import java.beans.PropertyDescriptor;
7 import java.lang.reflect.Method;
8
9 public class TestBean {
10
11 public static void main(String[] args) throws Exception {
12 testProDes();
13 testBeanInfo();
14 testBeanInfo1();
15
16 }
17
18 static void testBeanInfo() throws Exception{
19 Dog d = (Dog) Class.forName("reflect.Dog").newInstance();
20 String myname = "name";
21 BeanInfo beanInfo = Introspector.getBeanInfo(Dog.class);//beaninfo中存放bean的所有信息
22 PropertyDescriptor[] propertyDescriptors = beanInfo.getPropertyDescriptors();
23 for(PropertyDescriptor pd : propertyDescriptors){
24 if(pd.getName().equals(myname)){
25 Method m1 = pd.getWriteMethod();
26 m1.invoke(d,"zhangsan");
27 Method m2 = pd.getReadMethod();
28 m2.invoke(d);
29 System.out.println(d);
30 break;
31 }
32 }
33 }
34
35 static void testProDes() throws Exception{
36 Dog d = (Dog) Class.forName("reflect.Dog").newInstance();
37 String myname = "name" ;//这里是根据函数名确定的string的变量名,而不是内部定义的私有属性名。
38 PropertyDescriptor pd = new PropertyDescriptor(myname, Dog.class);//propertyDescriptor直接得到属性描述符
39
40 Method m1 = pd.getWriteMethod();
41 m1.invoke(d,"zhangsan");
42 Method m2 = pd.getReadMethod();
43 m2.invoke(d);
44 System.out.println(d);
45 }
46 static void testBeanInfo1() throws Exception{
47 Dog d = (Dog) Class.forName("reflect.Dog").newInstance();
48 String myname = "name";
49 BeanInfo beanInfo = Introspector.getBeanInfo(Dog.class);//beaninfo中存放bean的所有信息
50 MethodDescriptor[] mds = beanInfo.getMethodDescriptors();
51 for(MethodDescriptor md : mds){
52 String str = md.getName();
53 if (str.matches("[gs]et.*")) {
54 System.out.println(str);//打印set/get开头的函数
55 }
56 }
57 }
58
59 }
60
61 class Dog {
62 private String n ;
63 private int a ;
64
65 public Dog(){
66 }
67 public Dog(String name,int age){
68 this.n = name;
69 this.a = age;
70 }
71
72 public String getName() {
73 return n;
74 }
75 public void setName(String name) {
76 this.n = name;
77 }
78 public int getAge() {
79 return a;
80 }
81 public void setAge(int age) {
82 this.a = age;
83 }
84 @Override
85 public String toString(){
86 return n + " : "+a;
87 }
88
89 }
在这里我们故意将Dog类中的get/set方法和属性名定义的不一样。但是通过内省,我们并不需要知道确切的内部属性的名字,通过get/set的名字,就可以实现对属性的设置和读取。
第一个函数直接用propertyDescriptor得出属性描述符,然后通过read方法求出get对应的method,write方法求出set对应的method,通过method的反射invoke函数实现对属性的操作。
第二种函数和第一个函数实现一样的功能,但是路径不一样,先用introspector静态方法得出beaninfo,beaninfo中包含了所有的关于类的信息,再通过beaninfo的方法得出propertyDescriptor,在按照第一种的实现方法实现对属性的操作。
第三个函数顺便测试了methoddescripor,将这个javabean类中的所有get/set的方法打印出来。