小菜提到的實體類,即項目中業務或者數據庫表的映射,貌似也可以稱為模型,不同的語言中叫法不同吧!!
舉個例子,比如在某個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"));)。
注意事項:
使用本類肯定會降低程序效率,慎重使用。
本類只支持帶有一個參數的屬性賦值方法。
如果多個實體類中有重復的方法名稱,則默認使用的是在集合中靠前的那個實體類的方法。
寫在后面的話:
本文只是提供一種思路,肯定不是最好的解決方案,也不一定能滿足讀者的需求,高手勿噴。。。
附:完整演示代碼