在 Spring Data 中为同一 QueryDSL 路径创建多个别名

我有一个通用的Spring Data存储库接口,可以扩展,允许我自定义查询执行。我正在尝试将内置的基本相等性测试扩展到默认存储库实现中,以便我可以使用Spring Data REST执行其他查询操作。例如:QuerydslBinderCustomizer

GET /api/persons?name=Joe%20Smith  // This works by default
GET /api/persons?nameEndsWith=Smith  // This requires custom parameter binding.

我遇到的问题是,我创建的实体路径的每个别名似乎都会覆盖前面的别名绑定。

@NoRepositoryBean
public interface BaseRepository<T, ID extends Serializable>
    extends PagingAndSortingRepository<T, ID>, QueryDslPredicateExecutor<T>, QuerydslBinderCustomizer { 

    @Override
    @SuppressWarnings("unchecked")
    default void customize(QuerydslBindings bindings, EntityPath entityPath){

        Class<T> model = entityPath.getType();
        Path<T> root = entityPath.getRoot();
        for (Field field: model.getDeclaredFields()){
            if (field.isSynthetic()) continue;
            Class<?> fieldType = field.getType();
            if (fieldType.isAssignableFrom(String.class)){
                // This binding works by itself, but not after the next one is added
                bindings.bind(Expressions.stringPath(root, field.getName()))
                        .as(field.getName()  + "EndsWith")
                        .first((path, value) -> {
                            return path.endsWith(value);
                        });
                // This binding overrides the previous one
                bindings.bind(Expressions.stringPath(root, field.getName()))
                        .as(field.getName()  + "StartsWith")
                        .first((path, value) -> {
                            return path.startsWith(value);
                        });
            }
        }
    }
}

是否可以为同一字段创建多个别名?这可以通过通用方式完成吗?


答案 1

您可以通过以下方式创建绑定到 QueryDSL 的瞬态属性:

@Transient
@QueryType(PropertyType.SIMPLE)
public String getNameEndsWith() {
    // Whatever code, even return null
}

如果您使用的是 QueryDSL 注释处理器,您将在元数据 Qxxx 类中看到“nameEndsWith”,因此您可以像绑定任何持久化属性一样绑定它,但不需要持久化它。


答案 2

推荐