使用 mongoTemplate 进行分页

Posted

技术标签:

【中文标题】使用 mongoTemplate 进行分页【英文标题】:Pagination with mongoTemplate 【发布时间】:2015-05-15 20:02:52 【问题描述】:

我有一个带有 Pageable 的查询:

Query query = new Query().with(new PageRequests(page, size))

如何使用 MongoTemplate 执行它?我没有看到返回 Page<T> 的单一方法。

【问题讨论】:

【参考方案1】:

MongoTemplate 没有返回Page 的方法。 find() 方法返回一个普通的 List

with(new PageRequests(page, size) 在内部使用 MongoDB 查询来调整 skiplimit(我认为是由计数查询进行的)

Page 可以与 MongoDB repositories 结合使用,MongoDB repositories 是 Spring 数据存储库的一个特例。

因此,您必须使用MongoRepositoryPage findAll(Pageable pageable) 来获得分页结果(实际上继承自PagingAndSortingRepository)。

【讨论】:

【参考方案2】:

MongoTemplate 确实没有带有 Pageables 的 findXXX

但您可以使用 Spring Repository PageableExecutionUtils

在您的示例中,它看起来像这样:

Pageable pageable = new PageRequests(page, size);
Query query = new Query().with(pageable);
List<XXX> list = mongoTemplate.find(query, XXX.class);
return PageableExecutionUtils.getPage(
                       list, 
                       pageable, 
                       () -> mongoTemplate.count(Query.of(query).limit(-1).skip(-1), XXX.class));

就像在原始 Spring Data Repository 中一样,PageableExecutionUtils 将执行计数请求并将其包装成一个漂亮的 Page 供您使用。

Here你可以看到spring也在做同样的事情。

【讨论】:

PageableExecutionUtils 类在哪里? 来自 Spring Data commons:github.com/spring-projects/spring-data-commons/blob/master/src/… 这不会导致查询在 MongoDB 端运行两次并增加网络开销吗? @checklist 是的,查询运行了两次。由于查询应该针对索引集合运行,因此应该没什么大不了的。好的,如果您是 facebook 或 twitter,这可能是一件大事;D。我没有其他解决方案 这很有趣,但是如果您的查询返回数十万或数百万行不带分页,当它试图将所有这些文档读入内存时,这将削弱您的服务器。使用聚合框架要好得多【参考方案3】:

基于 d0x 的回答并查看 spring code。我正在使用这种变体,它可以消除 spring-boot-starter-data-mongodb 依赖项,而无需添加 spring data commons。

@Autowired
private MongoOperations mongoOperations;

@Override
public Page<YourObjectType> searchCustom(Pageable pageable) 
    Query query = new Query().with(pageable);
    // Build your query here

    List<YourObjectType> list = mongoOperations.find(query, YourObjectType.class);
    long count = mongoOperations.count(query, YourObjectType.class);
    Page<YourObjectType> resultPage = new PageImpl<YourObjectType>(list , pageable, count);
    return resultPage;

【讨论】:

仅供参考:为了获得正确的计数值 - 查询应该重置:跳过,限制值。计数调用应为:mongoOperations.count(query.skip(-1).limit(-1), YourObjectType.class)。否则它将返回不正确的查询计数。【参考方案4】:
return type Mono<Page<Myobject>>...

return this.myobjectRepository.count()
        .flatMap(ptiCount -> 
          return this.myobjectRepository.findAll(pageable.getSort())
            .buffer(pageable.getPageSize(),(pageable.getPageNumber() + 1))
            .elementAt(pageable.getPageNumber(), new ArrayList<>())
            .map(ptis -> new PageImpl<Myobject>(ptis, pageable, ptiCount));
        );

【讨论】:

【参考方案5】:

spring mongo模板默认没有分页查找的方法。它搜索并返回整个记录列表。我试过这个,它工作:

Pageable pageable = new PageRequests(0, 10);                              
Query query = new Query(criteria); 
query.with(pageable);   
List<User> lusers = mt.find(query, User.class);   
Page<User> pu = new PageImpl<>(lusers, pageable, mongoTemplate.count(newQuery(criteria), User.class));

【讨论】:

【参考方案6】:

此处提供的解决方案均不适用于我自己的情况。 我尝试在下面的中型帖子中使用此解决方案,它从未返回分页结果,但返回所有结果,这不是我所期望的

return PageableExecutionUtils.getPage(
        mongoTemplate.find(query, ClassName.class),
        pageable,
        () -> mongoTemplate.count(query.skip(0).limit(0), ClassName.class)
);

所以我找到了一个更好的方法来解决这个问题,它在我的情况下很有效:

return PageableExecutionUtils.getPage(
            mongoTemplate.find(query.with(pageable), ClassName.class),
            pageable,
            () -> mongoTemplate.count(query, ClassName.class));

【讨论】:

以上是关于使用 mongoTemplate 进行分页的主要内容,如果未能解决你的问题,请参考以下文章

mongoTemplate聚合操作Demo

SpringBoot系列之MongoTemplate加PageHelper分页实现

mongotemplate怎么获得统计数组的大小

MongoTemplate 排序不适用于 LocalDateTime

怎么打印mongotemplate的日志

使用 Spring Boot 和 MongoTemplate 对 MongoDB 结果进行排序