首页 > 解决方案 > JPQL:OR 不会按预期工作

问题描述

我在存储库中定义了以下方法:

@Query("SELECT t FROM Treatment t WHERE " +
        " (t.promotionCode.promotion.id=:promotionId)  " +
        " order by t.id desc")
Page<Treatment> findByPromotionId(@Param("promotionId")Integer id, Pageable pr);

它按预期工作:我得到了一个具有属于给定促销的促销代码的治疗列表。

但是后来我需要添加第二个促销代码,因此一个治疗最多可以链接到两个促销(两个促销代码可能属于同一个促销,这不是问题)。所以我尝试将新要求添加到查询中:

@Query("SELECT t FROM Treatment t WHERE " +
            " (t.promotionCode.promotion.id=:promotionId)  " +
            " OR " +
            " (t.promotionCode2.promotion.id=:promotionId)  " +
            " order by t.id desc")
Page<Treatment> findByPromotionId(@Param("promotionId")Integer id, Pageable pr);

但我不会工作。生成的 SQL 是

select ...
from treatment treatment0_ 
  cross join promotion_code promotionc1_ 
  cross join promotion_code promotionc2_ 
where 
  treatment0_.promotion_code_id=promotionc1_.id and
  treatment0_.promotion_code2_id=promotionc2_.id and 
  (promotionc1_.promo_id=? or promotionc2_.promo_id=?)
order by 
  treatment0_.id desc limit ?

如您所见,只要其中一个促销代码为空,则不满足条件。

一些细节,即使它们从代码中很明显:

我想通过任何促销代码列查找与促销相关的所有治疗。这两个字段都可以为空。

我该如何解决这个问题?

标签: javapostgresqlspring-bootjpajpql

解决方案


您可以尝试使用条件 API。

CriteriaBuilder 接口提供了工厂方法,它们接受两个 Expression 操作数(包括 Predicate 实例)并返回一个新的 Predicate 实例:

  Predicate p1 = cb.and(isInUN, isInEU);  // Member of both UN and EU
  Predicate p2 = cb.or(isInOECD, isLarge); // Either OECD member or large

其他工厂方法可用于不同数量的谓词:

  Predicate p3 = cb.and(p1, isLarge, cb.isTrue(isInOECD));
  Predicate p4 = cb.or(p2, cb.isTrue(isInUN), cb.isTrue(isInEU));

在上面的代码中,非 Predicate 布尔表达式使用 isTrue 方法转换为 Predicate 实例。这是必需的,因为在非二进制版本中,工厂方法只接受 Predicate 实例作为参数。

源网址:https ://www.objectdb.com/java/jpa/query/jpql/logical#Criteria_Query_Logical_Operators_


推荐阅读