反射機制Reflection 簡單的理解就是對class類的運用,在項目當中,適用於很多獨特的場景,比如我們項目中的需求,1 我這邊有兩條數據,我主要知道兩條數據哪些字段做了變更;2 我們跟別的webServer做聯調,適配推送過去的字段等等。
1 幾個注意核心類和方法
- Field : 提供有關類或接口的單個字段的信息和動態訪問。
getName() :返回此Field對象表示的字段的名稱.
getType(): 返回一個Class對象,該對象標識此Field對象表示的字段的聲明類型 。- getDeclaredField(String name) : 返回一個Field對象。
- getDeclaredFields() :返回一個Field數組
- 當然用的的不止這些,具體看業務和Class 文檔
2 生成公共類
public class Conversion {
//將model中與entity中類型和名稱相同的屬性值賦值給對應的entity的屬性,並返回entity
public static <T1, T2> T2 TypeConversion(T1 model, T2 entity) {
List<Map<String, Object>> modelList = getFiledInfo(model);
List<Map<String, Object>> entityList = getFiledInfo(entity);
for (Map e : entityList) {
for (Map m : modelList) {
/**
* 判斷類型和屬性名是否都相同
*/
if (e.get("type").toString().equals(m.get("type").toString()) && e.get("name")
.toString()
.equals(m.get("name").toString())) {
try {
Field f = entity.getClass().getDeclaredField(e.get("name").toString());
f.setAccessible(true);
f.set(entity, m.get("value"));
} catch (Exception ex) {//查看其父類屬性
try {
Field f = entity.getClass().getSuperclass().getDeclaredField(e.get("name").toString());
f.setAccessible(true);
f.set(entity, m.get("value"));
} catch (Exception e1) {
logger.error(ex.getMessage(), "conversion類型轉換錯誤 " + ex);
}
}
}
}
}
return entity;
}
/**
* 根據屬性名獲取屬性值
*/
private static Object getFieldValueByName(String fieldName, Object o) {
try {
String firstLetter = fieldName.substring(0, 1).toUpperCase();
String getter = "get" + firstLetter + fieldName.substring(1);
Method method = o.getClass().getMethod(getter, new Class[]{});
Object value = method.invoke(o, new Object[]{});
return value;
} catch (Exception e) {
logger.error(e.getMessage(), e);
return null;
}
}
/**
* 獲取屬性名數組
*/
private static String[] getFiledName(Object o) {
Field[] fields = o.getClass().getDeclaredFields();
String[] fieldNames = new String[fields.length];
for (int i = 0; i < fields.length; i++) {
fieldNames[i] = fields[i].getName();
}
return fieldNames;
}
/**
* 獲取屬性類型(type),屬性名(name),屬性值(value)的map組成的list
*/
private static List<Map<String, Object>> getFiledInfo(Object o) {
List<Map<String, Object>> list = new ArrayList<>();
List<Field> fields = new ArrayList<>();
fields.addAll(Arrays.asList(o.getClass().getDeclaredFields()));
/**
* 如果存在父類,獲取父類的屬性值,類型,名稱並添加到一起
*/
Class sc = o.getClass().getSuperclass();
if (sc != null) {
fields.addAll(Arrays.asList(sc.getDeclaredFields()));
}
for (Field field : fields) {
Map<String, Object> infoMap = new HashMap<>();
infoMap.put("type", field.getType().toString());
infoMap.put("name", field.getName());
infoMap.put("value", getFieldValueByName(field.getName(), o));
list.add(infoMap);
}
return list;
}
}
3 項目實踐:
- 3.1需求1 :我這邊有兩條數據,我想知道知道兩條數據哪些字段做了變更【通過遍歷,查出字段name相同的字段的value是否相同,記錄下來,返回到List<Map<String, Object>> 集合中,接下來你就可以拿着這個結合做各種業務處理了】
// 這邊比較數據的值 public List<Map<String, Object>> getCompareData(HecYbMedCodeSyncEntity entityHis, HecYbMedCodeSyncEntity entityNow) { List<Map<String, Object>> lists = new ArrayList<>(); List<Map<String, Object>> modelList = Conversion.getFiledInfo(entityHis); List<Map<String, Object>> entityList = Conversion.getFiledInfo(entityNow); modelList.forEach(mapHis -> { entityList.forEach(mapNow -> { Map<String, Object> map = new HashMap<>(); //這邊需要把獲取得到的字段名 跟 醫保中心的字段名做個匹配,把我們這邊的字段名轉成他們那邊的字段名 if ( mapHis.get("name").toString().equals(mapNow.get("name").toString())) {//判斷名字相同 if (!(mapHis.get("value").toString().equals(mapNow.get("value").toString())) ) {//判斷值不相同 且 反射實體不爲空 map.put("bgzdbm", fieldMappedEnum.getSourceFieldName()); //變更字段編碼 map.put("bgzdmc", fieldMappedEnum.getDesc()); //變更字段名稱 map.put("bgqz00", mapHis.get("value").toString()); //變更前值 map.put("bghz00", mapNow.get("value").toString()); //變更後值 map.put("bgsj00", DateUtil.limitDay(entityNow.getSyncTime())); //變更時間 lists.add(map); } } }); });
- 3.2 我們跟別的webServer做聯調,我需要把我的數據庫字段的名稱改成webServer上的值。
我的設計思路是,設計一個枚舉,把我數據庫的字段A,和webServer接口上的字段對應上B,這樣給B賦值的時候就可以通過枚舉獲取得到A,並且通過A就可以獲取到A的值,這樣就繞過了人工的給B賦值的繁瑣操作。
調用如下:
FieldMappedEnum fieldMappedEnum = FieldMappedEnum.findSourceFieldByTarget(mapHis.get("name").toString());
map.put("bgzdbm", fieldMappedEnum.getSourceFieldName()); //變更字段編碼
map.put("bgzdmc", fieldMappedEnum.getDesc()); //變更字段名稱
....