如何在运行时为嵌套属性创建CriteriaBuilder表达式?

时间:2019-06-03 14:36:04

标签: java jpa criteria criteria-api

我正在尝试使用Criteria API创建用于深度过滤的通用方法。

例如,如果要过滤“ id.person.name”属性或3个以上的嵌套属性。

我已经可以使用2个嵌套属性的过滤器,例如:“ person.name”。

下面的elseif条件正在使用3个嵌套属性,但是我正在尝试创建一种方法,如果出现新情况,则无需更改。

private <T> List<Predicate> filterInDepth(Map<String, String> params, Root<T> root,
        CriteriaBuilder criteriaBuilder, String field, Class<T> clazz)
        throws NoSuchFieldException {
    List<Predicate> predicates = new ArrayList<>();
    String[] compositeFields = field.split(REGEX_FIELD_SPLITTER);
    if (compositeFields.length == 2) {
        Join<Object, Object> join = root.join(compositeFields[0]);
        predicates.add(criteriaBuilder.equal(join.get(compositeFields[1]),
                params.get(field)));
    }
    else if (clazz.getDeclaredField(compositeFields[0]).getType()
            .getDeclaredField(compositeFields[1]).getType()
            .getDeclaredField(compositeFields[2]) != null) {
        predicates.add(criteriaBuilder.equal(root.get(compositeFields[0])
                .get(compositeFields[1]).get(compositeFields[2]), params.get(field)));
    }
    return predicates;
}

所以我试图拆分字符串,循环并创建一个表达式。但是我无法连接表情。

1 个答案:

答案 0 :(得分:0)

我找到了一种解决方案,希望对某人有所帮助。

private <T> List<Predicate> filterInDepth(Map<String, String> params, Root<T> root,
        CriteriaBuilder criteriaBuilder, String field, Class<T> clazz)
        throws NoSuchFieldException {
    List<Predicate> predicates = new ArrayList<>();
    String[] compositeFields = field.split(REGEX_FIELD_SPLITTER);

    if (isNestedFieldExists(clazz, compositeFields)) {
        Path path = null;
        for (String part : compositeFields) {
            if (path == null) {
                path = root.get(part);
            }
            else {
                path = path.get(part);
            }
        }

        predicates.add(criteriaBuilder.equal(path, params.get(field)));
    }

    return predicates;
}

private boolean isNestedFieldExists(Class clazz, String[] fieldChain)
        throws NoSuchFieldException {
    for (String field : fieldChain) {
        Field f = clazz.getDeclaredField(field);
        if (f.getType() == null) {
            return false;
        }

        clazz = f.getType();
    }

    return true;
}