mybatis foreach collection详解(转)

本文详细介绍了MyBatis中foreach标签的使用方法,包括不同情况下collection属性的设置规则,以及如何处理单个和多个参数的情况。

以下内容转自:http://www.cnblogs.com/aigeileshei/p/6109355.html

在SQL开发过程中,动态构建In集合条件查询是比较常见的用法,在Mybatis中提供了foreach功能,该功能比较强大,它允许你指定一个集合,声明集合项和索引变量,它们可以用在元素体内。它也允许你指定开放和关闭的字符串,在迭代之间放置分隔符。这个元素是很智能的,它不会偶然地附加多余的分隔符。下面是一个演示示例:

  <select id="findByIdsMap" resultMap="BaseResultMap">
    Select
    <include refid="Base_Column_List" />
     from jria where ID in
     <foreach item="item" index="index" collection="list" 
                    open="(" separator="," close=")">
                   #{item}
            </foreach>
 </select> 

但由于官方文档对这块的使用,描述的比较简短,细节上也被忽略掉了(可能是开源项目文档一贯的问题吧),也使用不少同学在使用中遇到了问题。特别是foreach这个函数中,collection属性做什么用,有什么注意事项。由于文档不全,这块只能通过源代码剖析的方式来分析一下各个属性的相关要求。

collection属性的用途是接收输入的数组或是List接口实现。但对于其名称的要求,Mybatis在实现中还是有点不好理解的,所以需要特别注意这一点。

下面开始分析源代码(笔记使用的是Mybatis 3.0.5版本)

先找到Mybatis执行SQL配置解析的入口

MapperMethod.Java类中 public Object execute(Object[] args) 该方法是执行的入口.

针对in集合查询,对应用就是 selectForList或SelctForMap方法。

但不管调用哪个方法,都会对原来JDK传入的参数 Object[]类型,通过 getParam方法转换成一个Object,那这个方法是做什么的呢?分析源码如下:


 

上图中标红的两处,很惊讶的发现,一个参数与多个参数的处理方式是不同的(后续很多同学遇到的问题,就有一大部分出自这个地方)。如果参数个数大于一个,则会被封装成Map, key值如果使用了Mybatis的 Param注解,则会使用该key值,否则默认统一使用数据序号,从1开始。这个问题先记下,继续分析代码,接下来如果是selectForList操作(其它操作就对应用相应方法),会调用DefaultSqlSession的public List selectList(String statement, Object parameter, RowBounds rowBounds) 方法

又一个发现,见源代码如下:


 

上图标红部分,对参数又做了一次封装,我们看一下代码

 


 

现在有点清楚了,如果参数类型是List,则必须在collecion中指定为list, 如果是数据组,则必须在collection属性中指定为 array.

现在就问题就比较清楚了,如果是一个参数的话,collection的值取决于你的参数类型。

如果是多个值的话,除非使用注解Param指定,否则都是数字开头,所以在collection中指定什么值都是无用的。下图是debug显示结果。


 

针对上面分析的结果,下面给出了一个使用的解决方案,希望对大家对帮助。

在使用这个功能是需要特别注意以下规则:
1. 当查询的参数只有一个时 
  findByIds(List<Long> ids)
 1.a 如果参数的类型是List, 则在使用时,collection属性要必须指定为 list
 <select id="findByIdsMap" resultMap="BaseResultMap">
         Select
         <include refid="Base_Column_List" />
         from jria where ID in
                  <foreach item="item" index="index" collection="list" 
                         open="(" separator="," close=")">
                        #{item}
                </foreach>
  </select> 
 
 findByIds(Long[] ids)
 1.b 如果参数的类型是Array,则在使用时,collection属性要必须指定为 array
  <select id="findByIdsMap" resultMap="BaseResultMap">
                 select
                 <include refid="Base_Column_List" />
          from jria where ID in
                  <foreach item="item" index="index" collection="array" 
                         open="(" separator="," close=")">
                        #{item}
                </foreach>
  </select> 
 
2. 当查询的参数有多个时,例如 findByIds(String name, Long[] ids)
 这种情况需要特别注意,在传参数时,一定要改用Map方式, 这样在collection属性可以指定名称
         下面是一个示例
         Map<String, Object> params = new HashMap<String, Object>(2);
        params.put("name", name);
         params.put("ids", ids);
        mapper.findByIdsMap(params);
 
 <select id="findByIdsMap" resultMap="BaseResultMap">
                 select
                 <include refid="Base_Column_List" />
          from jria where ID in
                  <foreach item="item" index="index" collection="ids" 
                         open="(" separator="," close=")">
                        #{item}
                </foreach>
   </select> 
 
 
完整的示例如下:
例如有一个查询功能,Mapper接口文件定义如下方法:
List<Jria> findByIds(Long... ids);
使用 in 查询的sql拼装方法如下:
 <select id="findbyIds" resultMap="BaseResultMap">
                 select
                 <include refid="Base_Column_List" />
          from jria where ID in
                  <foreach item="item" index="index" collection="array" 
                         open="(" separator="," close=")">
                        #{item}
                </foreach>
  </select>
 3.如果参数为对象时,例如 deletes(Query query);query的数据封装在map中,以Query的属性名为健名,假设Query类
的结构为:
 class Query{
	Integer[] ids;
	String name;
 }
foreach 要取query对象里的ids数组时,在collection中直接取ids名作为循环对象:
 <foreach item="id" collection="ids">
	...
 </foreach>
 
  

<think>嗯,用户想了解MyBatisforEach标签的用法,需要详细的使用方法和示例。首先,我得回忆一下MyBatisforeach标签的基本结构和参数。根据引用的内容,foreach标签用于遍历集合,生成动态SQL,比如IN语句或者批量插入。 用户可能已经知道一些基础,但需要具体的示例和参数说明。比如collection属性应该对应传入的参数名称,item是每个元素的别名,还有open、separator、close这些属性控制循环生成的内容结构。引用里的示例代码显示,foreach可以放在where条件里生成多个OR条件,或者用逗号分隔的列表。 然后要分步骤讲解,可能先介绍基本语法,再讲参数含义,接着给出不同场景的例子,比如IN查询、批量插入、动态条件组合。每个例子需要结合代码片段,并且说明各个参数的作用。比如在批量插入时,使用foreach包裹插入的值,用逗号分隔。 还要注意用户可能的疑问点,比如collection属性的不同情况,当参数是List或者数组时,名称的写法可能不同。比如直接传入List的话,collection应该用"list",而如果是Map中的某个key,就要用对应的名称。这点需要明确说明,避免用户混淆。 另外,常见应用场景也很重要,用户可能想知道什么时候该用foreach,比如批量操作、动态IN子句、多条件查询。需要结合引用中的信息,说明这些场景下的具体使用方法。 最后要确保示例代码的正确性,比如XML中的标签语法是否正确,参数占位符是否正确使用#{}。同时,提醒用户注意SQL注入的问题,虽然MyBatis的#{}是预编译的,但使用foreach时还是要注意集合参数的合法性。 可能还需要对比一下不同的循环结构,比如在批量插入和条件查询中的不同配置,separator的使用区别,以及open和close如何包裹整个循环生成的内容。比如在IN子句中,open是"(",close是")",而separator是",",这样生成的SQL就是(值1, 值2, 值3)。 总结下来,回答的结构应该是:基本语法、参数说明、不同场景的示例、常见应用场景,以及注意事项。这样用户能逐步了解foreach的用法,并且知道如何应用到实际开发中。</think>### MyBatis中`<foreach>`标签的用法详解 `<foreach>`标签是MyBatis中用于动态生成SQL语句的核心标签之一,**主要用于遍历集合参数并拼接SQL片段**,在批量操作、动态条件查询等场景中广泛应用。以下是具体用法和示例: --- #### 一、基本语法与参数 ```xml <foreach collection="集合参数名" item="遍历元素别名" index="遍历索引别名" open="起始符号" separator="分隔符" close="结束符号"> #{item} </foreach> ``` - **`collection`**:必填,指定需遍历的集合参数名,若参数直接为`List`类型则用`list`,若为Map中的某个键则需对应键名[^4]。 - **`item`**:必填,定义集合中每个元素的引用名。 - **`open`/`close`**:定义循环体开始和结束时的符号(如`(`或`IN`语句的括号)。 - **`separator`**:定义元素间的分隔符(如逗号`,`或`OR`)。 --- #### 二、常见应用场景与示例 1. **动态IN查询** 根据用户ID列表查询用户信息: ```xml <select id="selectUsersByIds" resultType="User"> SELECT * FROM user WHERE id IN <foreach collection="ids" item="id" open="(" separator="," close=")"> #{id} </foreach> </select> ``` - 若传入`ids = [1,2,3]`,生成的SQL为:`WHERE id IN (1,2,3)`[^3]。 2. **批量插入数据** 插入多条用户记录: ```xml <insert id="batchInsertUsers"> INSERT INTO user (name, email) VALUES <foreach collection="userList" item="user" separator=","> (#{user.name}, #{user.email}) </foreach> </insert> ``` - 若传入`userList`包含3个对象,生成的SQL为:`VALUES (a,a@test.com), (b,b@test.com), (c,c@test.com)`[^1]。 3. **动态组合条件** 多条件筛选(如根据用户名或邮箱列表查询): ```xml <select id="selectByConditions" resultType="User"> SELECT * FROM user <where> <foreach collection="names" item="name" open="AND (" separator="OR" close=")"> name = #{name} </foreach> <foreach collection="emails" item="email" open="AND (" separator="OR" close=")"> email = #{email} </foreach> </where> </select> ``` - 生成的SQL可能为:`WHERE (name = 'Alice' OR name = 'Bob') AND (email = 'a@test.com')`[^3]。 --- #### 三、注意事项 1. **集合参数类型** - 若接口方法参数为`List`,`collection`需设置为`list`。 - 若参数为`Map`中的某个键,例如`map.put("ids", list)`,则`collection="ids"`。 2. **性能优化** 批量操作时需控制单次处理的数据量(如分批次插入),避免SQL过长导致数据库性能下降。 3. **SQL注入** MyBatis通过`#{}`预编译机制防止注入,但需确保传入的集合参数合法[^2]。 ---
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值