在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>
相关推荐
Mybatis批量foreach merge into的用法是通过Mybatis的动态SQL语法foreach循环插入实现的,这种方法可以批量插入时间价格表数据。如果某个套餐的某天的价格存在,则更新,不存在则插入。下面是该方法的详细介绍: ...
在MyBatis中,`<foreach>`标签是一个非常重要的元素,它主要用于动态SQL语句的构建,尤其是在处理集合数据类型如List、Array、Map时。`<foreach>`标签的使用可以极大地提高代码的可读性和可维护性,避免了传统的字符...
首先,让我们先了解一下foreach标签的基本使用方法。foreach标签是MyBatis提供的一种批量插入数据的方式,可以将一个集合中的数据批量插入到数据库中。foreach标签的基本语法如下: ```xml insert into table_...
"狂神说Mybatis"的课堂笔记涵盖了这个框架的核心概念、配置、使用方法及实战技巧,旨在帮助学习者深入理解和应用Mybatis。 1. **Mybatis简介**:Mybatis 是由Apache基金会维护的一个开源项目,它解决了传统JDBC编程...
3. 在使用MyBatis foreach collection时,需要特别注意collection属性的名称,MyBatis在实现中对collection属性的名称有特殊要求。 通过对MyBatis foreach collection的分析,我们可以更好地使用该功能,避免常见的...
7. **动态SQL**:Mybatis的动态SQL功能非常强大,可以使用`if`, `choose`, `when`, `otherwise`, `trim`, `where`, `set`, `foreach`等标签编写条件语句,使得SQL更具灵活性。 8. **缓存**:Mybatis提供了本地缓存...
"Mybatis foreach标签带来的空格、换行...Mybatis 中的 foreach 标签带来的空格、换行、回车问题可以通过使用 replace 函数或 trim 函数来解决。这两个函数都可以删除空格、换行、回车符号,从而生成正确的 SQL 语句。
5. **动态SQL**:介绍如何使用`if`、`choose`(`when`、`otherwise`)、`foreach`等元素实现条件判断和循环,进行灵活的SQL构建。 6. **MyBatis与POJOs**:解释如何将数据库表与Java对象进行映射,以及自动类型转换...
除了XML配置,MyBatis也支持在Java类和方法上使用注解进行映射,简化开发流程。 5. **MyBatis的Executor执行器** MyBatis提供了三种Executor执行器:SimpleExecutor、ReuseExecutor和BatchExecutor,分别对应简单...
4. **MyBatis注解使用**:除了XML方式外,MyBatis还支持注解方式来定义Mapper接口中的方法,简化配置,提高开发效率。 5. **MyBatis的动态SQL**:通过`<if>`, `<choose>`, `<when>`, `<otherwise>`, `<where>`, `...
因此,当Mybatis尝试通过`foreach`访问`key`和`value`时,它找不到对应的方法,因为Integer类没有这些属性。 为了解决这个问题,我们需要确保在使用`foreach`时,传入的集合与我们的映射逻辑相匹配。如果是Map,...
- 除了XML方式,Mybatis还支持注解配置,可以直接在Mapper接口的方法上使用注解声明SQL。 5. **参数映射与结果映射** - **参数映射**:Mybatis通过Map、POJO或注解来映射传入参数,实现动态SQL。 - **结果映射**...
4. **动态SQL**:MyBatis的动态SQL功能强大,可以在XML映射文件或注解中使用if、choose、when、otherwise、where、trim、foreach等标签,实现灵活的SQL构建,避免硬编码SQL。 5. **参数映射**:MyBatis支持简单参数...
《玩转》Java系列之Mybatis学习手册,旨在帮助开发者深入理解并熟练运用Mybatis这一流行持久层框架。本手册将全面解析Mybatis的核心概念、关键功能及其在实际开发中的应用,助你从初学者晋升为Mybatis高手。 1. **...
Mybatis 的强大之处在于其动态 SQL 功能,可以使用 if、choose、when、otherwise、where、foreach 等标签编写条件语句,使 SQL 更具灵活性。 9. **缓存机制** Mybatis 内置了本地缓存和二级缓存,可以提高查询...
总的来说,正确理解和使用MyBatis的`<foreach>`标签是避免此类错误的关键。在编写动态SQL时,仔细检查参数类型、属性名称以及`<foreach>`标签的配置,可以有效地防止此类问题的发生。同时,阅读MyBatis的源码也能...
MyBatis 中的 foreach Collection 用法小结(三种) MyBatis 中的 foreach 语句是用来迭代一个集合,以便在 SQL 语句中生成相应的条件语句。foreach 语句的主要用途是在构建 in 条件中,例如 select * from blog ...
5. **MyBatis 动态 SQL**:动态 SQL 是 MyBatis 的一大亮点,它允许在 XML 映射文件中使用一系列标签(如 trim|where|set|foreach|if|choose|when|otherwise|bind)来构建动态 SQL,实现条件判断和 SQL 拼接。...
在"MyBatis学习笔记.zip"这个压缩包中,包含了四天的学习内容,分别以"MyBatis_day1"、"MyBatis_day2"、"MyBatis_day3"和"MyBatis_day4"命名,这些笔记可能详细阐述了MyBatis的核心概念和使用方法。让我们逐一解析...
11. **MyBatis 动态 SQL**:利用 if、choose、when、otherwise、trim、where、foreach 等标签实现灵活的 SQL 生成。 12. **MyBatis 执行流程**:了解 MyBatis 如何解析 XML 或注解,生成 PreparedStatement,执行 ...