通用 criteriaUpdate set boolean = !boolean

Generic criteriaUpdate set boolean = !boolean

我想使用 criteriaUpdate 创建这样的更新查询:

UPDATE <SOME TABLE>
SET SELECTED = !SELECTED
WHERE
[DYNAMIC QUERY HERE]

我能得到的最接近的代码是:

public <T> Query createRevertSelectionQuery(Class<T> clazz, EntityManager em, Specification<T> s) {
    CriteriaBuilder cb = em.getCriteriaBuilder();
    CriteriaUpdate<T> criteriaUpdate = cb.createCriteriaUpdate(clazz);
    Root<T> root = criteriaUpdate.from(clazz);
    Predicate p = cb.and(new Predicate[] {s.toPredicate(root, null, cb)});
    Expression<Boolean> e =cb.not((root.get("selected").as(Boolean.class)));
    Path<Boolean> selected = root.get("selected");
    criteriaUpdate.set(selected, e);
    criteriaUpdate.where(p);
    Query q = em.createQuery(criteriaUpdate);       
    return q;       
}

但它失败了,因为我收到以下查询:

update com.redknee.suspense.mgt.model.Moc as generatedAlias0 
set generatedAlias0.selected = generatedAlias0.selected <> true 
where 
[dynamic query]

报错

org.hibernate.hql.internal.ast.QuerySyntaxException:意外标记:<> 第 1 行第 118 列附近

有人可以帮忙吗?

我不确定这是一个错误还是不应该以这种方式使用。

在 where 子句中,NOT 和任何其他操作数都非常有效。但是,无论您尝试什么,Hibernate 查询构建器似乎总是会优化那些括号(在我看来,始终使用括号可能仍然是一个好习惯,但这只是一个意见)。

强制使用括号的一种方法是使用 JPA Subquery。请参见下面的示例。请注意,我根据自己的喜好略微更改了 JPA 对象名称,未包含 Specification,因为它与此解决方案无关:

CriteriaBuilder cb = em.getCriteriaBuilder();
CriteriaUpdate<T> update = cb.createCriteriaUpdate(clazz);
Root<T> from = update.from(clazz);
Path<Boolean> selected = from.get("selected");

// Subquery just "joins back" to the same row and
// returns a negated boolean value of "selected" from the original row
Subquery<Boolean> subSelect = update.subquery(Boolean.class);
Root<T> subFrom = subSelect.from(clazz);
subSelect.select(cb.not(selected));
subSelect.where(cb.equal(from.get("id"), subFrom.get("id")));

update.set(selected, subSelect);