Mybatis游标查询大量数据

背景

对大量数据进行处理时,为防止内存泄漏情况发生,所以采用mybatis plus游标方式进行数据查询处理,当查询百万级的数据的时候,使用游标可以节省内存的消耗,不需要一次性取出所有数据,可以进行逐条处理或逐条取出部分批量处理

mapper层

  1. 使用Cursor类型进行数据接收
  2. @Options,fetchSize设置为Integer最小值
  3. @Select,写查询sql
@Options(resultSetType = ResultSetType.FORWARD_ONLY, fetchSize = Integer.MIN_VALUE)
@Select("select domain from illegal_domain where icpstatus != #{icpstatus}")
Cursor<IllegalDomain> getDayJobDomain(@Param("icpstatus") Integer icpstatus);

service层

Cursor<IllegalDomain> domainList = illegalDomainMapper.getDayJobDomain(1);

数据处理

  1. forEach方式
domainList.forEach(illegalDomain -> {
    //处理逻辑,根据业务需求自行完成
    Future<IcpStatusVo> future = checkIcpThreadPool.submit(new IcpCheckThread(illegalDomain.getDomain(), configMap));
    results.add(future);
});
  1. 迭代器
Iterator<IllegalDomain> iter = domainList.iterator();

while (iter.hasNext()) {

    <!--// Fetch next 10 employees-->
    <!--for(int i = 0; i<10 && iter.hasNext(); i++) {-->
    <!--    smallChunk.add(iter.next());-->
    <!--}-->
    
    //处理逻辑,根据业务需求自行完成
    Future<IcpStatusVo> future = checkIcpThreadPool.submit(new IcpCheckThread(illegalDomain.getDomain(), configMap));
    results.add(future);
}

资源释放

使用完毕后,在finally块释放资源,否则游标不关闭也可能会导致内存溢出问题

try{
    //your code
    
} catch (Exception e) {
    log.error(e);
} finally {
    if(null != domainList){
        try {
            domainList.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}
  • 3
    点赞
  • 28
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
MyBatis是一个Java持久层框架,它提供了一种使用简单的方式来执行SQL查询、插入、更新和删除操作。在MyBatis中,游标可以用于处理大量数据查询结果集,以避免一次性加载所有数据造成的内存消耗过大。 在MyBatis中使用游标的步骤如下: 1. 定义游标类型的结果集映射:在MyBatis的映射文件(例如XML文件)中,使用`<resultMap>`标签定义游标类型的结果集映射。例如: ```xml <resultMap id="cursorResultMap" type="yourEntity"> <!-- 定义结果集映射 --> </resultMap> ``` 2. 使用游标类型的返回值类型:在执行查询操作时,将返回值类型指定为游标类型。例如,在映射文件中定义一个带有游标类型返回值的查询语句: ```xml <select id="yourQuery" resultMap="cursorResultMap"> <!-- 查询语句 --> </select> ``` 3. 处理游标结果集:在Java代码中,通过调用MyBatis的`select`方法执行查询操作,并通过游标逐行处理结果集。例如: ```java try (SqlSession sqlSession = sqlSessionFactory.openSession()) { try (Cursor<YourEntity> cursor = sqlSession.selectCursor("yourNamespace.yourQuery")) { while (cursor.hasNext()) { YourEntity entity = cursor.next(); // 处理每一行数据 } } } ``` 在上面的代码中,`selectCursor`方法返回的`Cursor`对象可用于逐行获取结果集,`hasNext`方法用于判断是否还有下一行数据,`next`方法用于获取下一行数据。 需要注意的是,游标类型的查询只适用于支持游标数据库,如Oracle、PostgreSQL等。对于MySQL来说,MyBatis并不直接支持游标类型的查询,因为MySQL本身并不提供游标的机制。 希望这些信息能够帮助到你!如果还有其他问题,请随时提问。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值