Spring Data JPA - JpaRepository 中的自定义排序
Posted
技术标签:
【中文标题】Spring Data JPA - JpaRepository 中的自定义排序【英文标题】:Spring Data JPA - Custom Sort in JpaRepository 【发布时间】:2016-01-23 22:18:39 【问题描述】:我正在将 Spring Data JPA 与 Spring Data REST 一起使用,并且我为我的 Thing
实体创建了一个 JpaRepository。
@Repository
public interface ThingRepository extends JpaRepository<Thing, Long>
@RestResource(path = "findByName", rel = "findByName")
Page findByName(@Param("name") String name, Pageable page);
我想应用排序。我想根据自定义评级算法对Thing
列表进行排序。
List<Thing> sortByRating(List<Thing> things)
// custom logic
return things;
;
如果可能的话,我希望使用自定义函数来对结果进行排序。 Spring JPA 中这种事情的最佳方法是什么?如何让我的存储库使用我的函数对结果集进行排序?
【问题讨论】:
【参考方案1】:您可以自定义您的对象页面(Pageable)。如果你得到请求的数据:
public PageRequest customSort(Args...)
String[] newParams = "newParam";
Sort sort = page.getSort();
Sort.Order dataSort =
sort.iterator().next();
Sort newSort = Sort.by(dataSort.getDirection(), newParams);
return PageRequest.of(page.getPageNumber(), this.getPageSize(), newSort);
SQL 之前:
order by oldParam ASC/DESC;
SQL 之后:
order by newParam ASC/DESC;
【讨论】:
【参考方案2】:Spring Data 提供基本排序,如 here 所示。您也可以使用@Query
注释您的查询并使用order by
,但我相信您需要的是更复杂的排序逻辑。
如果您使用 Java 8,您可以使用它的功能对列表进行排序,this 文章展示了很好的示例,这需要更多的工作,但看起来就像您所追求的那样。请注意,为了从stream
转换为list
,您必须在stream
的末尾包含.collect(Collectors.toList());
。
示例(摘自文章):
employees.stream().sorted((Employee e1, Employee e2) -> e1.getHireDate()
.compareTo(e2.getHireDate())).collect(Collectors.toList());
编辑:我没有注意到你在使用分页。我有一个类似的问题,我必须在返回 Page
对象之前对页面内容做一些事情,所以我最终创建了 PageImpl<T>
类的扩展:
public class PageImplBean<T> extends PageImpl<T>
private static final long serialVersionUID = 1L;
private int number;
private int size;
private int totalPages;
private int numberOfElements;
private long totalElements;
private boolean previousPage;
private boolean first;
private boolean nextPage;
private boolean last;
private List<T> content;
@JsonIgnore
private Sort sort;
public PageImplBean()
super(new ArrayList<T>());
public PageImplBean(Page pagina)
super(new ArrayList<T>());
this.number = pagina.getNumber();
this.size = pagina.getSize();
this.totalPages = pagina.getTotalPages();
this.numberOfElements = pagina.getNumberOfElements();
this.totalElements = pagina.getTotalElements();
this.previousPage = pagina.hasPrevious();
this.first = pagina.isFirst();
this.nextPage = pagina.hasNext();
this.last = pagina.isLast();
public int getNumber()
return number;
public void setNumber(int number)
this.number = number;
public int getSize()
return size;
public void setSize(int size)
this.size = size;
public int getTotalPages()
return totalPages;
public void setTotalPages(int totalPages)
this.totalPages = totalPages;
public int getNumberOfElements()
return numberOfElements;
public void setNumberOfElements(int numberOfElements)
this.numberOfElements = numberOfElements;
public long getTotalElements()
return totalElements;
public void setTotalElements(long totalElements)
this.totalElements = totalElements;
public boolean isPreviousPage()
return previousPage;
public void setPreviousPage(boolean previousPage)
this.previousPage = previousPage;
public boolean isFirst()
return first;
public void setFirst(boolean first)
this.first = first;
public boolean isNextPage()
return nextPage;
public void setNextPage(boolean nextPage)
this.nextPage = nextPage;
public boolean isLast()
return last;
public void setLast(boolean last)
this.last = last;
public List<T> getContent()
return content;
public void setContent(List<T> content)
this.content = content;
public Sort getSort()
return sort;
public void setSort(Sort sort)
this.sort = sort;
public PageImpl<T> pageImpl()
return new PageImpl<T>(getContent(), new PageRequest(getNumber(),
getSize(), getSort()), getTotalElements());
在您的情况下,重要的是 setContent
方法在基类中不存在。您可以将其与您的排序方法一起使用:
PageImplBean<Thing> page = //call your rest repository
List<Thing> pageContent = page.getContent();
page.setContent(sortByRating(pageContent));
【讨论】:
哦,我没有注意到他的查询中的Pageable pageable
,我的错!编辑我的答案。以上是关于Spring Data JPA - JpaRepository 中的自定义排序的主要内容,如果未能解决你的问题,请参考以下文章
spring-data-jpa 和 spring-boot-starter-data-jpa 的区别
spring-data详解之spring-data-jpa:简单三步快速上手spring-data-jpa开发