spring data jpa 分页页码起始为0的问题

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了spring data jpa 分页页码起始为0的问题相关的知识,希望对你有一定的参考价值。

RT Page接口的getNumber方法,索引是0开始的,但这样的可读性不好,每次显示页码还要加1,我想要以1开始(如/manage/page/1,这样取第一页)。
@RequestMapping("list")
public String list(@RequestParam(name="type", defaultValue="0")Integer type,
String sort, String keyword, Pageable pageable, Model model)
Page<PageViewDTO> list = pageviewService.list(type, keyword, sort, pageable);
model.addAttribute("pages", list);
return "list";

请问各位大神,是不是需要自己封装一个类来实现呢?还是有更简便的方法吗?

返回时,重写getNumber父类加1,接受参数时-1
public class PageNumber<T> extends PageImpl<T> implements Page<T>

/**
* 前台传参1为起始页修改为spring data jpa的0为起始页
* @param pageable
* @return
*/
public static Pageable pageRequest(Pageable pageable)
return new PageRequest(pageable.getPageNumber()-1,pageable.getPageSize(),pageable.getSort());


/**
* Constructor of @code PageImpl.
*
* @param content  the content of this page, must not be @literal null.
* @param pageable the paging information, can be @literal null.
* @param total    the total amount of items available. The total might be adapted considering the length of the content
*/
public PageNumber(List<T> content, Pageable pageable, long total) 
super(content, pageable, total);


/**
* 重写当前页,将当前页加1返回前台,spring data jpa起始页0加1后返回前台
* @return
*/
@Override
public int getNumber() 
return super.getNumber()+1;



public Page<PermissionEntity> findAll(Pageable pageable) 
Page<PermissionEntity> permissionEntityPage = permissionEntityRepository.findAll(pageable);
return new PageNumber<>(permissionEntityPage.getContent(),pageable,permissionEntityPage.getTotalElements());



@GetMapping("/permission")
@ResponseBody
public Page<PermissionEntity> permission(@PageableDefault Pageable pageable)
return permissionEntityService.findAll(PageNumber.pageRequest(pageable));

参考技术A 在分页符后的第一页击活页眉页角,设置页码选项为续前页或同前。

Spring Data JPA中带条件的分页查询

最新Spring Data JPA官方参考手册 Version 2.0.0.RC2,2017-07-25

https://docs.spring.io/spring-data/jpa/docs/2.0.0.RC2/reference/html/

JPA参考手册 (找了半天, 在线版的只找到这个)

https://www.objectdb.com/java/jpa

 

Spring Data JPA的Specification类, 是按照Eric Evans的《领域驱动设计》书中Specification的概念和语义来定义查询条件的API。

使用Spring Data JPA, 我们一般将自己的dao接口继承CrudRepository接口和JpaSpecificationExecutor接口, 由框架生成代理类来完成具体的调用, 而不用自己写daoImpl实现类, 因为这两个接口自带了很多方法, 如果我们写实现类会发现一上来就需要实现十来个方法, 比较麻烦。

其中CrudRepository接口主要负责增/删/改的操作, JpaSpecificationExecutor接口主要负责查询的操作, 另外, 框架还支持在dao接口的方法名上定义一些简单的语义来进行增删改查, 底层会对应地做具体实现。

那如何封装具体的查询条件呢?

在service层调用dao接口从JpaSpecificationExecutor继承的抽象查询方法, 它就会自动让你准备相关实参, 其中Specification对象就是经常用在条件查询的方法的一个形参, 也就是说, 封装查询条件的过程转移到service层了。

我们一般以匿名内部类的方式new一个Specification对象, 实现其中的toPredicate方法, 举个例子,

Specification<Person> specification = new Specification<Person>() {
            @Override
            public Predicate toPredicate(Root<Person> root, CriteriaQuery<?> criteriaQuery, CriteriaBuilder criteriaBuilder) {
            ...
};

其中, Predicate, Root, CriteriaQuery, CriteriaBuilder都是javax.persistence包中的接口, 方法的这3个实参由框架交给我们。

Predicate意思是"描述语", 就是我们封装完查询条件后要交给Specification一个清楚的描述, 要怎么组合sql语句去查询。

Root表示为泛型里的Person对象描述一个根位置, 可以从这个根位置去取该对象的属性, 以及属性的属性, 类似对象导航的意思, 比如要取Person地址属性的城市, 就可以root.get("address").get("city").as(String.class), 其返回值是一个Expression对象;

CriteriaQuery代表条件查询,主要提供where、group by、having、order by等。

CriteriaBuilder用于构造筛选条件,主要提供equal、and、or、lt、gt、between、like等, 以及获得CriteriaQuery、CriteriaUpdate、CriteriaDelete对象。构造每个筛选条件一般需要Expression类型作为实参, 可以通过Root对象调用get()方法得到。如果有多个筛选条件, 调用criteriaBuilder的and、or等方法连接起来, 一般是链式调用的形式。

举个简单的实际例子:

//带条件的分页查询, 根据person的first_name和last_name进行模糊查询
//为了直观, 假设两个字段都存在且不为空串, 省掉非空判断和对应的处理
public Page<Person> findSearch(Person person, int page, int size) {
Specification<Person> specification = new Specification<Person>() {
@Override
public Predicate toPredicate(Root<Person> root, CriteriaQuery<?> criteriaQuery, CriteriaBuilder criteriaBuilder) {
Predicate predicate1 = criteriaBuilder.like(root.get("first_name").as(String.class), "%"+person.getFirstName()+"%");
Predicate predicate2 = criteriaBuilder.like(root.get("last_name").as(String.class), "%"+person.getLastName()+"%");
       Predicate finalPredicate = criteriaBuilder.and(predicate1, predicate2);
       return finalPredicate;
}
};
PageRequest pageRequest = PageRequest.of(page-1, size);
return personDao.findAll(specification, pageRequest);
}

 

当然这都是JPQL的语法了, 很多开发者也经常在dao接口中直接写SQL语句来让框架查询, 使用起来感觉有点类似MyBatis, 会显得清爽很多, 只需定义一个抽象方法加上对应的注解@Modifying和@Query(value="sql语句", nativeQuery=true)即可, 它还有一个好处, 不用让封装查询条件这种事情跑到service层去。

 

以上是关于spring data jpa 分页页码起始为0的问题的主要内容,如果未能解决你的问题,请参考以下文章

spring data jpa 的分页以及条件查询

Spring Data Jpa: 分页和排序

Spring Data Jpa: 分页和排序

spring data jpa 分页查询

在 Spring Data JPA 中创建分页

Spring Data JPA native query 分页