Mybatis 的 foreach 为什么会出现性能问题

本文探讨了在Mybatis中使用`foreach`出现性能问题的原因,通过对比不同方式执行相同查询的耗时,展示了直接拼接SQL的显著优势。文章通过分析`foreach`的源码,解释了`foreach`在解析时会转化为`PreparedStatement`,导致大量占位符解析,影响性能。建议在处理大量数据时,避免使用`foreach`,而是选择在代码中拼接SQL并使用`<sql>`标签提升效率。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

背景

最近在做一个类似于综合报表之类的东西,需要查询所有的记录(数据库记录有限制),大概有1W条记录。该报表需要三个表的数据,也就是根据这 1W 个 ID 去执行查询三次数据库,其中,有一条查询 SQL 是自己写,其他两条是根据别人提供的接口进行查询。

刚开始的时候,没有多想,直接使用 in 进行查询,使用 Mybatis 的 foreach 语句;项目中使用的是 jsonrpc 来请求数据,在测试的时候,发现老是请求不到数据,日志抛出的是 jsonrpc 超时异常,继续查看日志发现,是被阻塞在上面的三条SQL查询中。

在以前分析 Mybatis 的源码的时候,了解到,Mybatis 的 foreach 会有性能问题,所以改了下 SQL,直接在代码中拼接SQL,然后在 Mybatis 中直接使用 # 来获取,替换 class 测试了下,果然一下子就能查询出数据。

前提

这里先不考虑使用 in 好不好,如何去优化 in,如何使用 exists 或 inner join 进行代替等,这里就只是考虑使用了 in 语句,且使用了 Mybatis 的 foreach 语句进行优化,其实 foreach 的优化很简单,就是把 in 后面的语句在代码里面拼接好,在配置文件中直接通过 #{xxx} 或 ${xxx} 当作字符串直接使用即可。

测试

在分析 foreach 源码之前,先构造个数据来看看它们的区别有多大。

建表语句:

CREATE TABLE person
(
    id int(11) PRIMARY KEY NOT NULL,
    name varchar(50),
    age int(11),
    job varchar(50)
);

插入 1W 条数据:

POJO 类:

@Getter
@Setter
@ToString
@NoArgsConstructor
@AllArgsConstructor
public class Person implements Serializable {
    private int id;
    private String name;
    private String job;
    private int age;
}

方式一

通过原始的方式,使用 foreach 语句:

1. 在 dao 里面定义方法:

List<Person> queryPersonByIds(@Param("ids") List<Integer> ids);

2. 配置文件SQL:

<select id="queryPersonByIds" parameterType="list" resultMap="queryPersonMap">
    select * from person where 1=1
    <if test="ids != null and ids.size() > 0">
        and id in
        <foreach collection="ids" item="item" index="index" separator="," open="(" close=")">
            #{item}
        </foreach>
    </if>
</select>

3. 执行 main 方法:

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = { "classpath:spring-mybatis.xml" })
public class MainTest {

    @Autowired
    private IPersonService personService;

    @Test
    public void test(){
        // 构造 1W 个 ID
        List<Integer> ids = new ArrayList<>();
        for (int i = 1; i <= 10000; i++) {
            ids.add(i);
        }
        long start = System.currentTimeMillis();

        // 执行三次
        personService.queryPersonByIds(ids);
        personService.queryPersonByIds(ids);
        personService.queryPersonByIds(ids);

        long end = System.currentTimeMillis();
        System.out.println(String.format("耗时:%d", end - start));
<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]。 ---
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值