我正在尝试使用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;
}
所以我试图拆分字符串,循环并创建一个表达式。但是我无法连接表情。
答案 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;
}