`
frank1998819
  • 浏览: 758218 次
  • 性别: Icon_minigender_1
  • 来自: 南京
文章分类
社区版块
存档分类

mybatis foreach *用法*(转)

 
阅读更多

在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集合查询,对应用就是 selectForListSelctForMap方法。

 

但不管调用哪个方法,都会对原来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批量foreach merge into的用法是通过Mybatis的动态SQL语法foreach循环插入实现的,这种方法可以批量插入时间价格表数据。如果某个套餐的某天的价格存在,则更新,不存在则插入。下面是该方法的详细介绍: ...

    测试mybatis里foreach用法

    在MyBatis中,`&lt;foreach&gt;`标签是一个非常重要的元素,它主要用于动态SQL语句的构建,尤其是在处理集合数据类型如List、Array、Map时。`&lt;foreach&gt;`标签的使用可以极大地提高代码的可读性和可维护性,避免了传统的字符...

    mybatis foreach批量插入数据:Oracle与MySQL区别介绍

    首先,让我们先了解一下foreach标签的基本使用方法。foreach标签是MyBatis提供的一种批量插入数据的方式,可以将一个集合中的数据批量插入到数据库中。foreach标签的基本语法如下: ```xml insert into table_...

    狂神说Mybatis.zip

    "狂神说Mybatis"的课堂笔记涵盖了这个框架的核心概念、配置、使用方法及实战技巧,旨在帮助学习者深入理解和应用Mybatis。 1. **Mybatis简介**:Mybatis 是由Apache基金会维护的一个开源项目,它解决了传统JDBC编程...

    详解mybatis foreach collection示例

    3. 在使用MyBatis foreach collection时,需要特别注意collection属性的名称,MyBatis在实现中对collection属性的名称有特殊要求。 通过对MyBatis foreach collection的分析,我们可以更好地使用该功能,避免常见的...

    Mybatis进阶详细用法可运行代码

    7. **动态SQL**:Mybatis的动态SQL功能非常强大,可以使用`if`, `choose`, `when`, `otherwise`, `trim`, `where`, `set`, `foreach`等标签编写条件语句,使得SQL更具灵活性。 8. **缓存**:Mybatis提供了本地缓存...

    Mybatis中foreach标签带来的空格\换行\回车问题及解决方案

    "Mybatis foreach标签带来的空格、换行...Mybatis 中的 foreach 标签带来的空格、换行、回车问题可以通过使用 replace 函数或 trim 函数来解决。这两个函数都可以删除空格、换行、回车符号,从而生成正确的 SQL 语句。

    mybatis学习手册和整合mybatis+spring

    5. **动态SQL**:介绍如何使用`if`、`choose`(`when`、`otherwise`)、`foreach`等元素实现条件判断和循环,进行灵活的SQL构建。 6. **MyBatis与POJOs**:解释如何将数据库表与Java对象进行映射,以及自动类型转换...

    mybatis高级视频教程

    除了XML配置,MyBatis也支持在Java类和方法上使用注解进行映射,简化开发流程。 5. **MyBatis的Executor执行器** MyBatis提供了三种Executor执行器:SimpleExecutor、ReuseExecutor和BatchExecutor,分别对应简单...

    day35_mybatis.zip

    4. **MyBatis注解使用**:除了XML方式外,MyBatis还支持注解方式来定义Mapper接口中的方法,简化配置,提高开发效率。 5. **MyBatis的动态SQL**:通过`&lt;if&gt;`, `&lt;choose&gt;`, `&lt;when&gt;`, `&lt;otherwise&gt;`, `&lt;where&gt;`, `...

    Mybatis foreach标签使用不当导致异常的原因浅析

    因此,当Mybatis尝试通过`foreach`访问`key`和`value`时,它找不到对应的方法,因为Integer类没有这些属性。 为了解决这个问题,我们需要确保在使用`foreach`时,传入的集合与我们的映射逻辑相匹配。如果是Map,...

    Mybatis系列学习源码

    - 除了XML方式,Mybatis还支持注解配置,可以直接在Mapper接口的方法上使用注解声明SQL。 5. **参数映射与结果映射** - **参数映射**:Mybatis通过Map、POJO或注解来映射传入参数,实现动态SQL。 - **结果映射**...

    mybatis-3.5.0.zip

    4. **动态SQL**:MyBatis的动态SQL功能强大,可以在XML映射文件或注解中使用if、choose、when、otherwise、where、trim、foreach等标签,实现灵活的SQL构建,避免硬编码SQL。 5. **参数映射**:MyBatis支持简单参数...

    学习手册 “玩转”Java系列 — Mybatis

    《玩转》Java系列之Mybatis学习手册,旨在帮助开发者深入理解并熟练运用Mybatis这一流行持久层框架。本手册将全面解析Mybatis的核心概念、关键功能及其在实际开发中的应用,助你从初学者晋升为Mybatis高手。 1. **...

    Mybatis的使用

    Mybatis 的强大之处在于其动态 SQL 功能,可以使用 if、choose、when、otherwise、where、foreach 等标签编写条件语句,使 SQL 更具灵活性。 9. **缓存机制** Mybatis 内置了本地缓存和二级缓存,可以提高查询...

    mybatis中foreach报错:_frch_item_0 not found的解决方法

    总的来说,正确理解和使用MyBatis的`&lt;foreach&gt;`标签是避免此类错误的关键。在编写动态SQL时,仔细检查参数类型、属性名称以及`&lt;foreach&gt;`标签的配置,可以有效地防止此类问题的发生。同时,阅读MyBatis的源码也能...

    mybatis 中 foreach collection的用法小结(三种)

    MyBatis 中的 foreach Collection 用法小结(三种) MyBatis 中的 foreach 语句是用来迭代一个集合,以便在 SQL 语句中生成相应的条件语句。foreach 语句的主要用途是在构建 in 条件中,例如 select * from blog ...

    MyBatis面试专题及答案.pdf

    5. **MyBatis 动态 SQL**:动态 SQL 是 MyBatis 的一大亮点,它允许在 XML 映射文件中使用一系列标签(如 trim|where|set|foreach|if|choose|when|otherwise|bind)来构建动态 SQL,实现条件判断和 SQL 拼接。...

    MyBatis学习笔记.zip

    在"MyBatis学习笔记.zip"这个压缩包中,包含了四天的学习内容,分别以"MyBatis_day1"、"MyBatis_day2"、"MyBatis_day3"和"MyBatis_day4"命名,这些笔记可能详细阐述了MyBatis的核心概念和使用方法。让我们逐一解析...

    mybatis视频教学-

    11. **MyBatis 动态 SQL**:利用 if、choose、when、otherwise、trim、where、foreach 等标签实现灵活的 SQL 生成。 12. **MyBatis 执行流程**:了解 MyBatis 如何解析 XML 或注解,生成 PreparedStatement,执行 ...

Global site tag (gtag.js) - Google Analytics