【问题标题】:Spring Data JPA - Custom Sort in JpaRepositorySpring Data JPA - 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 中这种事情的最佳方法是什么?如何让我的存储库使用我的函数对结果集进行排序?

【问题讨论】:

    标签: java spring jpa spring-data spring-data-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,我的错!编辑我的答案。
      猜你喜欢
      • 1970-01-01
      • 2013-11-23
      • 1970-01-01
      • 2015-12-07
      • 1970-01-01
      • 2017-04-04
      • 2015-06-06
      • 2014-06-01
      • 2016-09-10
      相关资源
      最近更新 更多