Spring Data JPA - JpaRepository中的自定义排序

9

我正在使用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中,这种情况的最佳方法是什么?我该如何使我的Repository使用我的函数来对结果集进行排序?

2个回答

0

Spring Data提供基本的排序,如here所示。您还可以使用@Query注释查询,并使用order by进行排序,但我认为您需要更复杂的排序逻辑。

如果您正在使用Java 8,则可以使用其功能对列表进行排序,this文章展示了良好的示例,需要更多的工作,但似乎是您想要的。请注意,为了从流转换为列表,您必须在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,我的错!请修改我的答案。 - dubonzi

0

您可以自定义对象页面(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;

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接