map接收参数 mybatis只做传入部分参数指定的update 使用foreach循环map产生动态sql

面对只传入部分参数更新单表的情况,使用MyBatis时若直接通过实体类会导致未传入的null字段被更新。本文探讨两种解决方案:1) 通过反射对比map与数据库实体,但存在反射效率低和类型转换复杂性;2) 利用MyBatis的foreach动态SQL,主要处理ArrayList到JSON字符串的转换,并注意剔除非更新字段如ID。

问题:

1、更新某单表,实体类有N个参数,但是只传入部分参数进行更新。

2、如果用实体类接收,则没有传的参数会被设置为null ,sql的XML如果用完整的update更新,则会认为进行了整体更新;

3、如果直接传入map进行更新,使用mybatis的 

<trim prefix="SET" suffixOverrides=",">
   <if test="@Ognl@isNotEmpty(createDate)">
      create_date = #{createDate} ,
   </if>
   <if test="@Ognl@isNotEmpty(modifyDate)">
      modify_date = #{modifyDate} ,
   </if>
   <if test="@Ognl@isNotEmpty(addr)">
      addr = #{addr} ,
   </if>
   <if test="@Ognl@isNotEmpty(bond)">
      bond = #{bond} ,
   </if>

这样的语句进行判断,则会出现一种情况: 当实际传入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 id="update2">
		UPDATE tbl_company SET
		<foreach collection="params.keys" item="key" separator="," >
			${key}=#{params[${key}]}
		</foreach>
		WHERE
		id = #{id}
	</update>

然后,封装接收到的 params,打包一层进map (其实此处需要剔除params的id的参数,否则把id也set了一遍)

再打包一层 id 作为参数。 

封装的新的map作为参数,丢进去循环。 

public HttpEntity update(Map<String,Object> 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<String, Object> camelToUnderline(Map<String, Object> map) {
        Map<String, Object> result = new HashMap<>();
        for (Map.Entry<String, Object> 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();
    }

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值