内省与JavaBean

概述

JavaBean代表一类特殊的Java类,这种类主要用来存储和传递属性信息,JavaBean中的方法主要用于设置和获取这些私有属性,他们有一定的命名规则,我们可以把它们想象为一个侧重属性信息的类,如JSP中的表单信息打包一样,这些信息就可以封装在一个JavaBean对象中,通过一些特殊的方式来操作。

Introspector,即为内省

public class Introspector
extends Object

Introspector 类为通过工具学习有关受目标 Java Bean 支持的属性、事件和方法的知识提供了一个标准方法。

简单说,Introspector就是一种操作JavaBean的标准方式

JavaBean一些简单操作

写一个Bean类用作测试

   1: public class Bean {

   2:     private String name;

   3:     private int age;

   4:     private int num;

   5:     private int score;

   6:     

   7:     public Bean(String name, int age, int num, int score) {

   8:         super();

   9:         this.name = name;

  10:         this.age = age;

  11:         this.num = num;

  12:         this.score = score;

  13:     }

  14:     public String getName() {

  15:         return name;

  16:     }

  17:     public void setName(String name) {

  18:         this.name = name;

  19:     }

  20:     public int getAge() {

  21:         return age;

  22:     }

  23:     public void setAge(int age) {

  24:         this.age = age;

  25:     }

  26:     public int getNum() {

  27:         return num;

  28:     }

  29:     public void setNum(int num) {

  30:         this.num = num;

  31:     }

  32:     public int getScore() {

  33:         return score;

  34:     }

  35:     public void setScore(int score) {

  36:         this.score = score;

  37:     }

  38: }

实现set和get方法的简单操作

   1: public class BeanTest {

   2:  

   3:     /**

   4:      * @param args

   5:      */

   6:     public static void main(String[] args) {

   7:         

   8:         //获取对象

   9:         Bean b = new Bean("Shawn",12,1,97);

  10:         //获取类字节码文件

  11:         Class beanClass = Bean.class;    

  12:         //设置需要操作的属性名

  13:         String propertyName = "name";

  14:         

  15:         //调用set和get方法

  16:         System.out.println(getProperty(beanClass, b, propertyName, null));

  17:         setProperty(beanClass, b, propertyName, new Object[]{"Feng"});

  18:         System.out.println(getProperty(beanClass, b, propertyName, null));

  19:                 

  20:     }

  21:     //封装javabean的set方法

  22:     private static void setProperty(Class beanClass,Object obj,String propertyName,Object[] values){

  23:         try {

  24:             //获取属性描述符对象

  25:             PropertyDescriptor pd = new PropertyDescriptor(propertyName, beanClass);

  26:             //得到set方法对象

  27:             Method setMethod = pd.getWriteMethod();

  28:             //利用反射执行set方法

  29:             setMethod.invoke(obj, values);

  30:         } catch (IllegalArgumentException e) {

  31:             // TODO Auto-generated catch block

  32:             e.printStackTrace();

  33:         } catch (IntrospectionException e) {

  34:             // TODO Auto-generated catch block

  35:             e.printStackTrace();

  36:         } catch (IllegalAccessException e) {

  37:             // TODO Auto-generated catch block

  38:             e.printStackTrace();

  39:         } catch (InvocationTargetException e) {

  40:             // TODO Auto-generated catch block

  41:             e.printStackTrace();

  42:         }

  43:     }

  44:     //封装javabean的get方法

  45:     private static Object getProperty(Class beanClass,Object obj,String propertyName,Object[] args){

  46:         Object retVal = null;

  47:         try {

  48:             //获取属性描述符对象

  49:             PropertyDescriptor pd = new PropertyDescriptor(propertyName, beanClass);

  50:             //得到get方法对象

  51:             Method getMethod = pd.getReadMethod();

  52:             //利用反射执行get方法

  53:             retVal = getMethod.invoke(obj, args);

  54:         } catch (IllegalArgumentException e) {

  55:             // TODO Auto-generated catch block

  56:             e.printStackTrace();

  57:         } catch (IntrospectionException e) {

  58:             // TODO Auto-generated catch block

  59:             e.printStackTrace();

  60:         } catch (IllegalAccessException e) {

  61:             // TODO Auto-generated catch block

  62:             e.printStackTrace();

  63:         } catch (InvocationTargetException e) {

  64:             // TODO Auto-generated catch block

  65:             e.printStackTrace();

  66:         }

  67:         return retVal;

  68:     }

  69: }

利用内省实现一些复杂操作

   1: public class BeanTest2 {

   2:  

   3:     /**

   4:      * @param args

   5:      * @throws IntrospectionException 

   6:      * @throws InvocationTargetException 

   7:      * @throws IllegalAccessException 

   8:      * @throws IllegalArgumentException 

   9:      */

  10:     public static void main(String[] args) throws IntrospectionException, IllegalArgumentException, IllegalAccessException, InvocationTargetException {

  11:         

  12:         //获取对象

  13:         Bean b = new Bean("Shawn",12,1,97);

  14:         //获取类字节码文件

  15:         Class beanClass = Bean.class;    

  16:         //设置需要操作的属性名

  17:         String propertyName = "name";

  18:         

  19:         //根据类字节码对象获取Bean信息

  20:         BeanInfo beanInfo = Introspector.getBeanInfo(beanClass);

  21:         //利用内省获取所有的属性描述符

  22:         PropertyDescriptor[] pds = beanInfo.getPropertyDescriptors();

  23:         //遍历所有的属性描述符,找到要操作的属性,进行操作

  24:         for(PropertyDescriptor pd : pds){

  25:             if(pd.getName().equals(propertyName)){

  26:                 Method setMethod = pd.getWriteMethod();

  27:                 Method getMethod = pd.getReadMethod();

  28:                 System.out.println(getMethod.invoke(b, null));

  29:                 setMethod.invoke(b, "Feng");

  30:                 System.out.println(getMethod.invoke(b, null));

  31:             }

  32:         }

  33:     }

  34: }

BeanUtils工具包

BeanUtils工具包是apach提供的一套非常方便的用于操作JavaBean的工具

BeanUtil实现set和get方法

   1: public class BeanUtilTest {

   2:  

   3:     /**

   4:      * @param args

   5:      * @throws NoSuchMethodException 

   6:      * @throws InvocationTargetException 

   7:      * @throws IllegalAccessException 

   8:      */

   9:     public static void main(String[] args) throws IllegalAccessException, InvocationTargetException, NoSuchMethodException {

  10:         

  11:         //获取对象

  12:         Bean b = new Bean("Shawn",12,1,97);

  13:         //设置需要操作的属性名

  14:         String propertyName = "score";

  15:         

  16:         //利用BeanUtils获取属性值

  17:         System.out.println(BeanUtils.getProperty(b, propertyName));

  18:         //利用BeanUtils设置属性值

  19:         BeanUtils.setProperty(b, propertyName, "33");

  20:         System.out.println(BeanUtils.getProperty(b, propertyName));

  21:     }

  22: }

我们可以看到,即便操作的是int型的属性,接收的仍然是字符串数据

BeanUtils包中还提供一个类PropertyUtils,也具有setProperty与getProperty的方法,与BeanUtils不同的是,传入的参数不需要进行类型转换,也就是说,如果BeanUtils中的方法类型转换出了问题时,可以采用PropertyUtils中的方法,直接传入对应类型的参数,同样可以达到目的。

同时,BeanUtils还支持属性的级联操作

比如一个人内部有心脏,而心脏内部又有左心房和右心房,那么我们可以直接通过BeanUtils设置和获取心脏的内部属性

   1: public class Heart{

   2:     private String left;

   3:     private String right;

   4:     public String getLeft() {

   5:         return left;

   6:     }

   7:     public void setLeft(String left) {

   8:         this.left = left;

   9:     }

  10:     public String getRight() {

  11:         return right;

  12:     }

  13:     public void setRight(String right) {

  14:         this.right = right;

  15:     }

  16: }

   1: public class Person{

   2:     private Heart heart = new Heart();

   3:     private Date birthday;

   4:     public Date getBirthday() {

   5:         return birthday;

   6:     }

   7:     public void setBirthday(Date birthday) {

   8:         this.birthday = birthday;

   9:     }

  10:     public Heart getHeart() {

  11:         return heart;

  12:     }

  13:     public void setHeart(Heart heart) {

  14:         this.heart = heart;

  15:     }

  16: }

   1: public class BeanUtilsTest2 {

   2:  

   3:     /**

   4:      * @param args

   5:      * @throws NoSuchMethodException 

   6:      * @throws InvocationTargetException 

   7:      * @throws IllegalAccessException 

   8:      */

   9:     public static void main(String[] args) throws IllegalAccessException, InvocationTargetException, NoSuchMethodException {

  10:         // TODO Auto-generated method stub

  11:         Person p = new Person();

  12:         System.out.println(BeanUtils.getProperty(p, "heart.left"));

  13:         BeanUtils.setProperty(p, "heart.left", "lll");

  14:         System.out.println(BeanUtils.getProperty(p, "heart.left"));

  15:     }

  16: }

还有一个重要的操作,BeanUtils可以将Map集合对象与JavaBean对象的对应属性内容进行转换

   1: public class BeanUtilsTest3 {

   2:  

   3:     /**

   4:      * @param args

   5:      * @throws InvocationTargetException 

   6:      * @throws IllegalAccessException 

   7:      * @throws NoSuchMethodException 

   8:      */

   9:     public static void main(String[] args) throws IllegalAccessException, InvocationTargetException, NoSuchMethodException {

  10:         

  11:         //将JavaBean的属性信息映射到一个Map集合中

  12:         Map map = new HashMap();

  13:         Bean b = new Bean("Shawn", 12, 14, 90);        

  14:         map = BeanUtils.describe(b);

  15:         System.out.println(map);

  16:         

  17:         //将map集合中的属性写入到JavaBean对象中

  18:         map = new HashMap();

  19:         map.put("name", "Feng");

  20:         map.put("score", 22);        

  21:         BeanUtils.populate(b, map);

  22:         System.out.println(b.toString());

  23:     }

  24: }

以上三点便是BeanUtils最常用的一些操作和特点,可以看到,BeanUtils的操作非常方便快捷,现在已经很普及了

JavaBean的命名规则

由于JavaBean中只有set和get方法是对外可见的,所以JavaBean内部属性用什么名字并不重要,只要设置好方法名称即可

一般规则是

如果方法名称为setXxx和getXxx,那么表示propertyName全为小写xxx

如果方法名称为setXXx或者setXXX,那么表示propertyName为对应的XXx以及XXX

一般setXxx以及setXXX比较常见,一个代表普通的变量,一个代表具有特殊含义的变量

所以我们写方法名称时尽量按照这个命名规则写,设置propertyName时使用相对应的名称即可

上一篇:shell脚本内与mysql交互


下一篇:解决VS2012新建MVC4等项目时,收到此模板加载程序集“NuGet.VisualStudio.Interop…”的错误