问题:
1、更新某单表,实体类有N个参数,但是只传入部分参数进行更新。
2、如果用实体类接收,则没有传的参数会被设置为null ,sql的XML如果用完整的update更新,则会认为进行了整体更新;
3、如果直接传入map进行更新,使用mybatis的
create_date = #{createDate} , modify_date = #{modifyDate} , addr = #{addr} , bond = #{bond} ,
这样的语句进行判断,则会出现一种情况: 当实际传入createDate = null 的时候,则会被过滤掉。比如,传入3个参数,其中一个是null,那么这样去组装SQL的时候,只会set两个值。
解决思路1:
首先还是需要用map接收,update必传的是ID,那么,首先,通过ID查一个数据库的实体A,然后遍历map,去和查到的实体A对比key,如果key是一样,则修改实体A的这个key对应的value。
问题:
(1) 需要用到JAVA的反射,动态通过key去反查value,(并且,反射的效率会比较低???)
(2)好不容易查到value了,可是map中的value是object,涉及到各种类型转换,比如,传入的是json字符串数组,则map默认为了 ArrayList,而实体类定义的很有可能是JSONArray。那么,此处也是可以能实现的,需要搞一个完整的java9大类型转换。
比如:
这样看来,这个解决方案,貌似不怎么滴,很麻烦。
解决思路2:
采用mybatis的 foreach来动态产生SQL。
当然,这里也涉及到类型转换,但是只需针对 ArrayList去转成 json字符串就好。
上代码:
首先,mybatis里的xml 按照foreach的方式来搞。 记住,此处传入的参数是 params 和 id
UPDATE tbl_company SET
${key}=#{params[${key}]}
WHERE
id = #{id}
然后,封装接收到的 params,打包一层进map (其实此处需要剔除params的id的参数,否则把id也set了一遍)
再打包一层 id 作为参数。
封装的新的map作为参数,丢进去循环。
public HttpEntity update(Map params) throws Exception {
//Company company = dao.selectOne("tbl_company.getById", param, Company.class);
params= CamelToUnderline.camelToUnderline(params);
Map map = new HashMap();
map.put("params",params);
map.put("id",params.get("id"));
dao.update("tbl_company.update2", map);
return new HttpEntity();
}
其次,用到了一点转换, 主要是解决数据库中驼峰问题,以及 JSON字符串数组进sql的问题。
如下:
public static final char UNDERLINE = '_';
/**
* 驼峰转下划线,此处在动态加载update更新语句 循环时可以用上 支持json字符串格式的录入
* @param map
* @return
*/
public static Map camelToUnderline(Map map) {
Map result = new HashMap<>();
for (Map.Entry entry : map.entrySet()) {
String newKey = camelToUnderline(entry.getKey());
Object objValue = entry.getValue();
if (entry.getValue() != null && entry.getValue().getClass().equals(ArrayList.class)) {
objValue = JSONObject.toJSONString(entry.getValue());
}
result.put(newKey, objValue);
}
return result;
}
/**
* 驼峰格式字符串转换为下划线格式字符串
*
* @param param
* @return
*/
private static String camelToUnderline(String param) {
if (param == null || "".equals(param.trim())) {
return "";
}
int len = param.length();
StringBuilder sb = new StringBuilder(len);
for (int i = 0; i < len; i++) {
char c = param.charAt(i);
if (Character.isUpperCase(c)) {
sb.append(UNDERLINE);
sb.append(Character.toLowerCase(c));
} else {
sb.append(c);
}
}
return sb.toString();
}