具有规范、分页和标准的 Spring Data JPA 存储库

Mac*_*zuk 7 java spring hibernate spring-data spring-data-jpa

我正在为实体列表实现搜索/过滤服务,使用具有规范和分页功能的 Spring Data JPA 存储库。我正在尝试减少查询次数(n+1 问题)并使用标准获取机制获取嵌套数据。

我有两个实体类:

@Entity
@Table(name = "delegations")
public class Delegation {

    @Id
    @GeneratedValue(strategy = IDENTITY)
    private Long id;

    @ManyToOne
    private Customer customer;

    // more fields, getters, setters, business logic...

}
Run Code Online (Sandbox Code Playgroud)

@Entity
@Table(name = "customers")
public class Customer {

    @Id
    @GeneratedValue(strategy = IDENTITY)
    private Long id;

    // more fields, getters, setters, business logic...
}
Run Code Online (Sandbox Code Playgroud)

DTO过滤器类:

public class DelegationFilter {

    private String customerName;

    // more filters, getters, setters...
}
Run Code Online (Sandbox Code Playgroud)

和搜索/过滤服务:

public class DelegationService {
    public Page<Delegation> findAll(DelegationFilter filter, Pageable page) {
        Specifications<Delegation> spec = Specifications.where(
                customerLike(filter.getCustomerName())
        );
        return delegationRepository.findAll(spec, page);
    }

    public List<Delegation> findAll(DelegationFilter filter) {
        Specifications<Delegation> spec = Specifications.where(
                customerLike(filter.getCustomerName())
        );
        return delegationRepository.findAll(spec);
    }

    private Specification<Delegation> customerLike(String customerName) {
        return (root, query, cb) -> {
            Join<Delegation,Customer> join = (Join) root.fetch(Delegation_.customer);
            return cb.like(cb.lower(join.get(Customer_.name)), addWildCards(customerName.toLowerCase()));
        };
    }

    private static String addWildCards(String param) {
        return '%' + param + '%';
    }
}
Run Code Online (Sandbox Code Playgroud)

问题

当我打电话时,findAll(DelegationFilter filter, Pageable page)我遇到了异常:

org.springframework.dao.InvalidDataAccessApiUsageException: 
org.hibernate.QueryException: query specified join fetching, but the owner 
of the fetched association was not present in the select list
Run Code Online (Sandbox Code Playgroud)

有没有办法解决这个问题?

findAll(DelegationFilter filter)(没有分页的方法)就像魅力一样......join只使用(没有fetch)也可以正常工作(即使有分页)

我知道 JPQL 有解决方案: Spring-Data FETCH JOIN with Paging is not working 但我想坚持使用标准 api ...

我正在使用 Spring Boot 1.4(spring 4.3.2,spring-data-jpa 1.10.2)和 Hibernate 5.0.9

gmc*_*gmc 9

我遇到了同样的问题,我找到了一个解决方法(source)。

您可以在运行时检查查询的返回类型,以便如果它是Long(计数查询返回的类型),则加入,否则获取。在您的代码中,它将如下所示:

...
private Specification<Delegation> customerLike(String customerName) {
    return (root, query, cb) -> {
        if (query.getResultType() != Long.class && query.getResultType() != long.class) {
          Join<Delegation,Customer> join = (Join) root.fetch(Delegation_.customer);
        } else {
          Join<Delegation,Customer> join = root.join(Delegation_.customer);
        }
        return cb.like(cb.lower(join.get(Customer_.name)), addWildCards(customerName.toLowerCase()));
    };
}
...
Run Code Online (Sandbox Code Playgroud)

我知道它不是很干净,但它是我使用 ATM 的唯一解决方案。

  • 我不敢相信这不是一个更常用的解决方案,我寻找了很长时间。我遇到的所有其他解决方案都需要单独的计数查询/JPQL,从不使用自定义规范。+100 非常感谢 (2认同)