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&lt;T&gt; 类的扩展:

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开发

spring data jpa怎么和solr整合

spring data jpa问题

Spring Data 系列 Spring+JPA(spring-data-commons)

spring-data-jpa软删除方案