微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

Java 规范 CriteriaBuilder 复杂查询

如何解决Java 规范 CriteriaBuilder 复杂查询

我在规范、构建器、查询方面的经验不是很丰富,我必须做一个像这样相当复杂的查询

select * from table where
   code in ('code1','code2' //codes) and
    (
        (
            date between "2020-03-23 //from" and "2020-03-30 //to"
            and
            status in ('Status1','Status2' //status)
        )
        or
        (
            date between "2021-03-23" and "2021-03-30"
            and
            status in ('Status3','Status4')
        )
    )

我有一个这样的 DTO:

public class SearchCriteria {

@Embedded
private Filters filters;

@Embeddable
@Getter
@Setter
public static class Filters {
    private List<String> codes;
    private List<TimePeriod> timePeriods;
}

@Embeddable
@Getter
@Setter
public static class TimePeriod {
    private List<String> status;
    private StartDate startDate;
}

@Embeddable
@Getter
@Setter
public static class StartDate {
    private LocalDate from;
    private LocalDate to;
}

这对我来说太难了。我正在尝试一切。我更愿意向您展示一个具体案例,以免引起误解。有人可以帮助我吗?我将不胜感激!

我不需要使用规范,我只需要能够重现那个查询示例,规范似乎是最好的选择。

谢谢大家。

解决方法

我认为您在正确的轨道上,标准类看起来不错。 以下是如何在方法中使用它来构建 JPA 条件并使用与您的实体对应的存储库执行查询:

public void query(List<String> codes,List<TimePeriod> timePeriods) {
    // build the code filter
    Specification<Table> codeSpec = (root,query,criteriaBuilder) -> {
        Path<String> codeField = root.get("code");
        var codePredicate = criteriaBuilder.in(codeField);
        codes.forEach(code -> codePredicate.value(code));
        return codePredicate;
    };
    // iterate over the time periods
    var timePeriodSpec = timePeriods.stream().map(timePeriod -> {
        Specification<Table> dateSpec = (root,criteriaBuilder) -> {
            Path<LocalDate> dateField = root.get("date");
            return criteriaBuilder.between(dateField,timePeriod.startDate.from,timePeriod.startDate.to);
        };
        Specification<Table> statusSpec = (root,criteriaBuilder) -> {
            Path<String> statusField = root.get("status");
            var statusPredicate = criteriaBuilder.in(statusField);
            timePeriod.status.forEach(status -> statusPredicate.value(status));
            return statusPredicate;
        };
        // combine the date and status filter
        return dateSpec.and(statusSpec);
    })
    .reduce(Specification::or).get(); // chain the time period filters together
    
    var fullSpec = codeSpec.and(timePeriodSpec);

    var result = tableRepository.findAll(fullSpec,Pageable.unpaged());
}

您还需要确保您的存储库实现了 JpaSpecificationExecutor 接口,但您可能已经想通了。

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。