解决实体类噩梦:联合实体类(Java反射+泛型实际应用)

       小菜提到的实体类,即项目中业务或者数据库表的映射,貌似也可以称为模型,不同的语言中叫法不同吧!!

       举个例子,比如在某个Web页面中,表单上有大量的数据需要提交,如果是初学者,很可能这样接收参数:

       String param1;String param2;String param3;……

       这样做的坏处很多,比如:代码会显得很乱,可能会出现大量重复代码,最主要的就是没有做到面向对象的“封装性”,导致程序不容易维护。

       由此,聪明的程序员们提出了实体类的概念,也就是用类来封装业务所需要的数据。

       public class User {

       private String uid;

       private String pwd;

       public String getUid() {

              return uid;

       }

       public void setUid(String uid) {

              this.uid = uid;

       }

       …..

       }

       这样一来,我们在保存数据时,只需创建一个对象,然后通过“.”的方式来访问对象的属性,提高扩展性、提高复用性、代码简洁等等好处不言而喻。

       在实际使用中,实体类往往是和某个业务或者数据库表相对应的,看起来很简单,但随着需求的复杂化,业务和业务之间可能会交叉,表和表之间可能会联合查询。

       这样一来,以前一一对应的实体类,便无法满足需求,因为某个实体类中可能找不到我们需要的属性,但是它却在另一个实体类中,而我们又不想随便在某个实体类中添加一个毫不相关的属性,因为这样做可能会打破类的职责单一原则。

       因此,我们不得不再新建一个实体类ClassAAndClassB,这个实体类包含了类A和类B的所有属性,对于稍有经验的程序员来说,这绝对是个噩梦,因为组合的可能性是在是太多,而且可能是很多个实体组合,等待我们的将是无数的实体类,导致程序混乱不堪。

       小菜一直苦于此事,今天终于通过Java的反射和泛型写了一个“联合实体类”。

       通过这个联合实体类,可以把任意多个实体类融合成一个实体类。

联合实体类代码:

  1 import java.lang.reflect.Method;
  2 import java.util.List;
  3 
  4 /**
  5  * 联合实体类
  6  * @author 杨元
  7  *
  8  */
  9 public class UniteEntity {
 10     
 11     //联合实体对象集合
 12     List<Object> entitys = null;
 13     //方法名称
 14     String fnName = "";
 15     //方法对象
 16     Method method = null;
 17     
 18     /**
 19      * 构造方法
 20      * @param entitys 需要联合的实体对象集合
 21      */
 22     public UniteEntity(List<Object> entitys){
 23         this.entitys = entitys;
 24     }
 25     
 26     /**
 27      * 获取某个取值方法
 28      * @param fnName 方法名称
 29      * @return 该取值方法返回值类型
 30      */
 31     public Class getFunction(String fnName){
 32         //保存方法名称
 33         this.fnName = fnName;
 34         //查找方法
 35         Method m = findMethod();
 36         //判断方法是否存在
 37         if(m != null){
 38             //获取目标方法的返回值类型
 39             Class type = m.getReturnType();
 40             //保存取值方法对象
 41             method = m;
 42             //返回该取值方法返回值类型
 43             return type;
 44         }else{
 45             return null;
 46         }
 47     }
 48     
 49     /**
 50      * 获取某个赋值方法
 51      * @param fnName 方法名称
 52      * @return 该赋值方法参数类型
 53      */
 54     public Class setFunction(String fnName){
 55         //保存方法名称
 56         this.fnName = fnName;
 57         //查找方法
 58         Method m = findMethod();
 59         //判断方法是否存在
 60         if(m != null){
 61             //获取目标方法的参数类型
 62             Class type = m.getParameterTypes()[0];
 63             //保存赋值方法对象
 64             method = m;
 65             //返回该赋值方法参数类型
 66             return type;
 67         }else{
 68             return null;
 69         }
 70     }
 71     
 72     /**
 73      * 调用某个方法,为属性赋值
 74      * @param <T> 赋值方法的参数类型
 75      * @param c
 76      * @param value 值内容
 77      */
 78     public <T> void setValue(Class<T> c,T value){
 79         //遍历实体类集合
 80         for(Object o : entitys){
 81             //出错继续执行
 82             try{
 83                 method.invoke(o, value);
            break;
84 }catch(Exception ex){} 85 } 86 } 87 88 /** 89 * 调用某个方法,取得属性的值 90 * @param <T> 取值方法的返回值类型 91 * @param c 92 * @return 取得值的内容 93 */ 94 public <T> T getValue(Class<T> c){ 95 //遍历实体类集合 96 for(Object o : entitys){ 97 //出错继续运行 98 try{ 99 //由于invoke返回的是Object类型,因此要强制转换成T类型 100 return (T)method.invoke(o); 101 }catch(Exception ex){} 102 } 103 return null; 104 } 105 106 /** 107 * 从实体对象集合中查找某个方法 108 * @return 方法对象 109 */ 110 private Method findMethod(){ 111 //遍历集合,寻找方法 112 for(Object o : entitys){ 113 //保证出错能继续运行 114 try{ 115 //获取对象所有公有方法 116 Method[] methods = o.getClass().getMethods(); 117 118 //遍历方法 119 for(Method m : methods){ 120 //匹配是否有目标方法 121 if(fnName.equals(m.getName())){ 122 //返回方法对象 123 return m; 124 } 125 } 126 }catch(Exception ex){} 127 } 128 return null; 129 } 130 }

 

调用方法:

 

 1 //创建一个对象集合
 2 List<Object> list = new ArrayList<Object>();
 3 
 4 //将需要融合的实体类填入集合
 5 list.add(new User());
 6 list.add(new Enterprise());
 7 
 8 //创建联合实体类对象
 9 UniteEntity ue = new UniteEntity(list);
10 
11 int i = 109;
12 
13 //调用实体类中方法名为setEnno的方法(赋值方法),并给一个参数i
14 ue.setValue(ue.setFunction("setEnno"), i);
15 //调用实体类中方法名为getEnno的方法(取值方法),并打印返回值
16 System.out.println(ue.getValue(ue.getFunction("getEnno")));

 

说明:

 

       用法很简单,创建联合实体对象的时候必须传入需要融合的实体对象集合。

       如果想调用的方法是取值方法,则先调用联合实体对象的getFunction方法,参数是方法的名称,一定要写对!!最好是复制!!这个步骤会查找到指定的方法,并且确定该方法的返回值类型,然后把getFunction方法的返回值作为参数,调用联合实体对象的getValue方法,即可取得属性值。

       如果想调用的方法是赋值方法,则先调用联合实体对象的setFunction方法,同理,该方法会确定参数的类型,把setFunction方法的返回值作为参数,调用联合实体对象的setValue方法,再加上需要赋给的值,即可给属性赋值。

       由于使用了泛型技术,所以本类比较安全、稳定。调用赋值方法时,如果传入的值和方法的参数类型不同,直接赋值失败,不会抛出异常;调用取值方法时,直接对取出来的值进行强制类型转换即可,无需验证数据类型(例如:int[] items = (int[])ue.getValue(ue.getFunction("getEmp"));)。

注意事项:

 

       使用本类肯定会降低程序效率,慎重使用。

       本类只支持带有一个参数的属性赋值方法。

       如果多个实体类中有重复的方法名称,则默认使用的是在集合中靠前的那个实体类的方法。

写在后面的话:

 

       本文只是提供一种思路,肯定不是最好的解决方案,也不一定能满足读者的需求,高手勿喷。。。

附:完整演示代码

 

       点我下载

 

 

posted @ 2013-04-19 00:16  杨元  阅读(16557)  评论(0编辑  收藏  举报