带有 oracle12c JSON_VALUE 函数的 JPA Criteria api

Ale*_*Bob 3 spring hibernate jpa spring-data-jpa spring-boot

我正在尝试使用 JSON_VALUE function() 来具体实现 Criteria API 查询是 Oracle12c:

select * from orderswhere (JSON_VALUE(duty, '$.department') in ( ? , ? , ? ,? , ? , ? , ?));
Run Code Online (Sandbox Code Playgroud)

模型:

@Entity
@Table(name = "orders")
@EntityListeners(AuditingEntityListener.class)
public class Trade implements Serializable {
     @Convert(converter = JpaConverterJson.class)
     private Map<String, Object> duty;
     .....
}

private Specification<InvoicePurchase> reportFormToSpec(Map<String, Object> 
    params) {
        return (root, query, cb) -> {
             List<Predicate> predicates = new ArrayList<>();
       if (!StringUtils.isEmpty(params.get("department"))) {
            Predicate predicate = cb.like(cb.function("json_value", String.class,
                    root.get("duty"),
                    cb.literal("$.department"))
                    , params.get("department"));
            predicates.add(predicate);
        }
            return query.where(predicates.toArray(new Predicate[predicates.size()])).getRestriction();
        };

    }
Run Code Online (Sandbox Code Playgroud)

但出现异常:ohengine.jdbc.spi.SqlExceptionHelper:ORA-40454:路径表达式不是文字

我在网上搜索了很长时间。但没有用。请帮助或尝试给出一些如何实现这一目标的想法。

错误代码:

   2017-11-23 12:26:55.271  WARN 11972 --- [nio-8080-exec-1] o.h.engine.jdbc.spi.SqlExceptionHelper   : SQL Error: 40454, SQLState: 99999
   2017-11-23 12:26:55.271 ERROR 11972 --- [nio-8080-exec-1] o.h.engine.jdbc.spi.SqlExceptionHelper   : ORA-40454: path expression not a literal
Run Code Online (Sandbox Code Playgroud)

Ist*_*vai 5

这不起作用的原因是,Hibernate 在后台会生成这样的 SQL:

SELECT * FROM T WHERE JSON_VALUE(?, ?)=?
Run Code Online (Sandbox Code Playgroud)

..这不适用于 Oracle。

解决方案是内联这些值。首先是一个小助手类:

public class HibernateInlineExpression extends LiteralExpression<String> {

  public HibernateInlineExpression(CriteriaBuilder criteriaBuilder, String literal) {
      super((CriteriaBuilderImpl) criteriaBuilder, literal);
  }

  @Override
  public String render(RenderingContext renderingContext) {
      return getLiteral();
  }
}
Run Code Online (Sandbox Code Playgroud)

使用上面的内容,您现在可以在谓词中执行 JSON 表达式:

cb.equal(cb.function("JSON_VALUE", String.class,
                new HibernateInlineExpression(cb, "JSON_COLUMN"),
                new HibernateInlineExpression(cb, "'$.my.json.property'")),
  new LiteralExpression<>((CriteriaBuilderImpl) cb, String.class, "somevalue"));
Run Code Online (Sandbox Code Playgroud)

当然上面是Hibernate特有的实现,EclipseLink等的用户必须想出类似的东西。请参阅我的关于从 JPA/Hibernate 使用 Oracle JSON 函数的博客文章,了解其他一些技巧。