为了账号安全,请及时绑定邮箱和手机立即绑定

仅当作为参数传递的 List 具有元素时才考虑 JPA 'where in'

仅当作为参数传递的 List 具有元素时才考虑 JPA 'where in'

开满天机 2021-06-27 12:39:27
我有一个应根据各种参数进行过滤的查询;这些参数之一是一个列表。如果列表中有条目,则应根据条目进行过滤;但如果列表为空/空,则不应对该字段进行任何过滤。我的想法是这样的:@Query("select a from Alert a where a.date >= :startDate " +            "and (((:countryIds) is null) or a.countryId in (:countryIds)) " +            "and (((:typeIds) is null) or a.siteTypeId in (:typeIds)) ")List<Alert> findBy(@Param("startDate") Date startDate,                   @Param("countryIds") Set<Long> countryIds,                   @Param("typeIds") Set<Long> typeIds);发送空列表会抛出 NPE;发送一个空列表它会生成以下SQL,这是无效的where alert0_.date >= '2018-01-01' and                                                        ((1, 123) is null or alert0_.countryId in (1, 123))我也尝试过使用 JPQL,and (((:countryIds) is empty) or a.countryId in (:countryIds))但是在尝试编译 JPQL(在应用程序启动时)时它也不起作用:Caused by: org.hibernate.hql.internal.ast.QuerySyntaxException: ??? is not mapped    at org.hibernate.hql.internal.ast.util.SessionFactoryHelper.requireClassPersister(SessionFactoryHelper.java:171) 或者使用 SpEL: "and (:#{countryIds.size() > 0} or (a.countryId in (:countryIds))) " 但同样,它不会编译 JPQL。我想到的唯一解决方案是动态生成丑陋的 JPQL 或填充所有现有值countryIds,siteTypeIds这是低效的。JPA 实现是 Hibernate,数据库是 MySQL。
查看完整描述

3 回答

?
ibeautiful

TA贡献1993条经验 获得超5个赞

我遇到了同样的问题,所以我使用嵌入式参数编写扩展解决方案


@Query("from PartPrice where "

            + "customer in :#{#customers} and "

            + "( (:#{#suppliers == null || #suppliers.size() == 0} = true and supplier is null) or (:#{#suppliers != null && #suppliers.size() > 0} = true and supplier in :#{#supplier}) ) and "

            + " productIdentifier.manufacturerId = :#{#productIdentifier.manufacturerId} and productIdentifier.productNumber = :#{#productIdentifier.productNumber} and "

            + " ( (:#{#isAbsPrice} = true and abs_price is not null) or (:#{#isAbsPrice} = false and abs_price is null) ) "

            + " and (validUntil is null or validUntil >= :#{#fromDate}) and (:#{#untilDate == null} = true or validFrom <= :#{#untilDate}) ")

其中供应商可为空、为空或包含值,而 productIdentifier 是嵌入的 id,其中包含 productNumber 和制造商 ID 传递为


@Param("productIdentifier") ProductIdentifier productIdentifier

间隔也是有效的,从 fromDate 到 null(永远)或 untilDate。


查看完整回答
反对 回复 2021-07-07
  • 3 回答
  • 0 关注
  • 167 浏览

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信