首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >Java规范CriteriaBuilder复杂查询

Java规范CriteriaBuilder复杂查询
EN

Stack Overflow用户
提问于 2021-04-29 17:48:06
回答 1查看 86关注 0票数 0

我对规范,构建器,查询不是很有经验,我必须像这样做一个相当复杂的查询:

代码语言:javascript
复制
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:

代码语言:javascript
复制
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;
}

这对我来说很难。我什么都试过了。我宁愿给你看一个具体的案例,以免引起误解。有人能帮帮我吗?非常感谢!

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

谢谢大家。

EN

回答 1

Stack Overflow用户

回答已采纳

发布于 2021-04-29 19:00:25

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

代码语言:javascript
复制
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, query, criteriaBuilder) -> {
            Path<LocalDate> dateField = root.get("date");
            return criteriaBuilder.between(dateField, timePeriod.startDate.from, timePeriod.startDate.to);
        };
        Specification<Table> statusSpec = (root, query, 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接口,但您可能已经弄清楚了这一点。

票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/67314943

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档