解決實體類噩夢:聯合實體類(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"));)。

注意事項:

 

       使用本類肯定會降低程序效率,慎重使用。

       本類只支持帶有一個參數的屬性賦值方法。

       如果多個實體類中有重復的方法名稱,則默認使用的是在集合中靠前的那個實體類的方法。

寫在后面的話:

 

       本文只是提供一種思路,肯定不是最好的解決方案,也不一定能滿足讀者的需求,高手勿噴。。。

附:完整演示代碼

 

       點我下載

 

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM