当前位置: 首页 > article >正文

mybatis使用foreach标签实现union集合操作

最近遇到一个场景就是Java开发中,需要循环多个表名,然后用同样的查询操作分别查表,最终得到N个表中查询的结果集合。在查询内容不一致时Java中跨表查询常用的是遍历表名集合循环查库,比较耗费资源,效率较低。在查询内容格式一致的情况下,便可以用Java的数据库连接工具模拟mysql中union的操作,这里我用的数据库交互工具是mybatis。

首先介绍下union:

在mysql中被称为集合操作,操作类型分为两种:UNION DISTINCT 和  UNION ALL;其中UNION和UNION DISTINCT是一样的功能。UNION功能为合并多个查询的结果并去重,UNION ALL的功能为合并多个查询的结果不去重。

这里mybatis在实现union操作时,用到的是foreach 标签,foreach标签用于循环语句,它很好的支持了数据和 List、set 接口的集合,并对此提供遍历的功能。最常用的地方就是对于一些 SQL 语句中含有 in 条件,或者批量处理数据需要迭代条件时,可以使用 foreach 。利用foreach这一迭代特性,也满足了union多个表的使用场景,整体思路框架代码如下:

实现层设置每一个union的查询sql的业务代码:

List<Map> queryCondition = new ArrayList<>();
//这里的namelIst代表存储联合查询表名的集合
for (Map nameMap: namelIst) {
            Map conditionMap = new HashMap();
            //查询内容
            String fields = "a.*,b.*";
            //拼接表名
            String tableName = "table_aaa a ," + nameMap.get("tableName") + "` b";
            //查询条件
            String joinsql = "a.nameId = b.id ";
            //将循环一次的查询条件,表名,内容放入一个集合中,当做union的一个查询部分
            conditionMap.put("fields", fields);
            conditionMap.put("tableName", tableName);
            conditionMap.put("joinsql", joinsql);
            //放入总的集合中作为传入mapper查询的条件
            queryCondition.add(conditionMap);
        }
        //执行查询
        List<Map> pages = this.baseMapper.getSelect(queryCondition);

传入的查询参数为一个list<map>集合。

/**
     * union查询示范
     * @param queryCondition
     * @return
     */
    List<Map> getSelect(@Param("queryCondition") List<Map> queryCondition);

xml文件的写法:

<select id="getSelect" resultType="java.util.Map">
        <foreach collection="queryCondition" item="condition" separator="union all">
            SELECT ${condition.fields}
            FROM ${condition.tableName}
            where ${condition.joinsql}
        </foreach>
    </select>

这里对foreach标签中的标签进行一下简单的总结:

foreach 标签有六个属性:item,index,collection,open, close,separator

属性 作用
item表示集合中每一个元素或子集合进行迭代循环时的别名
index    指定一个名字,表示在迭代过程中每次迭代到的位置
open    表示该语句以什么开始(如in 条件语句,以’('开始)
close    表示该语句以什么结束(如in 条件语句,以’)'结束。
separator    表示在每次进行迭代之间以什么符号作为分隔符(如in 条件语句,以‘,’作为分隔符)
collection    该属性是必选的,但在不同情况下该属性的值是不一样的,主要有以下 3 种情况:
(1)如果传入的是单参数且参数类型是一个 List,collection 属性值为 list
(2)如果传入的是单参数且参数类型是一个 array 数组,collection 的属性值为 array
(3)如果传入的参数是多个,需要把它们封装成一个 Map,当然单参数也可以封装成 Map。Map 的 key 是参数名,collection 属性值是传入的 List 或 array 对象在自己封装的 Map 中的 key。

通常在使用foreach标签时候,都是用来当做批量查询或者更新,或者where后面的条件使用,大概这样:

<foreach collection="各种集合" item="循环体中的别名" index="index"  open="(" separator="," close=")">
    自定义的各种参数
</foreach>

这里的操作属于mybatis的一个灵活运用,算是一个处理思路,鉴于网上现有相关思路不多,个人完成了测试,可以当做一种解决办法,提高一下程序的执行效率,避免多次连库。


http://www.kler.cn/a/134733.html

相关文章:

  • 【汇编】汇编编程中的指令传参方式
  • [微服务]redis数据结构
  • STM32特殊功能引脚详解文章·STM32特殊功能引脚能当作GPIO使用嘛详解!!!
  • 算法妙妙屋-------2..回溯的奇妙律动
  • T-SQL编程
  • redis缓存篇知识点总结
  • 如何利用1688批发市场价格做跨境代购业务?(API数据接口)
  • 软件测试/测试开发丨人工智能在软件测试领域的崭新前景
  • GPT-4充当“规划师、审计师”,颠覆性双层文生图表模型
  • openssl1.0.2版本Windows安装问题
  • 【系统稳定性】1.6 黑屏(三)
  • 如何远程控制别人电脑进行技术支持?
  • 金融知识普及月:别让“孤注一掷”照进现实,捷信反诈在行动
  • 使用Sqoop命令从Oracle同步数据到Hive,修复数据乱码 %0A的问题
  • 分布式下多节点WebSocket消息收发
  • 数字化时代,VR全景如何助力商企抢占市场份额?
  • 分布式原理
  • Python学习笔记(4)
  • CyclicBarrier使用案例
  • ⑩⑥ 【MySQL】详解 触发器TRIGGER,协助 确保数据的完整性,日志记录,数据校验等操作。
  • 微服务学习|Nacos配置管理:统一配置管理、配置热更新、配置共享、搭建Nacos集群
  • 2021年3月青少年软件编程(Python)等级考试试卷(一级)
  • ckplayer自己定义风格播放器的开发记录
  • SaleSmartly新增AI意图识别触发器!让客户享受更精准的自动化服务
  • C++sqrt函数题目
  • 聊一聊小程序单聊页面构思