JPA 将列表传递给命名本机查询中的 IN 子句
我知道我可以将列表传递给 JPA 中的命名查询,但是 NamedNativeQuery 怎么样?我尝试了很多方法,但仍然无法将列表传递给 NamedNativeQuery。有人知道如何将列表传递给 NamedNativeQuery 中的 in 子句吗?非常感谢!
NamedNativeQuery 如下:
@NamedNativeQuery(
name="User.findByUserIdList",
query="select u.user_id, u.dob, u.name, u.sex, u.address from user u "+
"where u.user_id in (?userIdList)"
)
其调用方式如下:
List<Object[]> userList = em.createNamedQuery("User.findByUserIdList").setParameter("userIdList", list).getResultList();
但是结果并不符合我的预期。
System.out.println(userList.size()); //output 1
Object[] user = userList.get(0);
System.out.println(user.length); //expected 5 but result is 3
System.out.println(user[0]); //output MDAVERSION which is not a user_id
System.out.println(user[1]); //output 5
System.out.println(user[2]); //output 7
I know I can pass a list to named query in JPA, but how about NamedNativeQuery? I have tried many ways but still can't just pass the list to a NamedNativeQuery. Anyone know how to pass a list to the in clause in NamedNativeQuery? Thank you very much!
The NamedNativeQuery is as below:
@NamedNativeQuery(
name="User.findByUserIdList",
query="select u.user_id, u.dob, u.name, u.sex, u.address from user u "+
"where u.user_id in (?userIdList)"
)
and it is called like this:
List<Object[]> userList = em.createNamedQuery("User.findByUserIdList").setParameter("userIdList", list).getResultList();
However the result is not as I expected.
System.out.println(userList.size()); //output 1
Object[] user = userList.get(0);
System.out.println(user.length); //expected 5 but result is 3
System.out.println(user[0]); //output MDAVERSION which is not a user_id
System.out.println(user[1]); //output 5
System.out.println(user[2]); //output 7
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(13)
在我的例子( EclipseLink , PostGreSQL )中,这有效:
同样在查询中不能使用 IN(?) 因为你会得到如下错误:
'IN(?)' 必须交换为 '= ANY(?)'
我的解决方案基于 Erhannis 概念。
In my case ( EclipseLink , PostGreSQL ) this works :
Also in query cannot use IN(?) because you will get error like :
'IN(?)' must be swapped to '= ANY(?)'
My solution was based on Erhannis concept.
在 JPA2 中尝试使用 Hibernate 作为提供者,看来 hibernate 确实支持接受“IN”列表并且它可以工作。 (至少对于命名查询来说,我相信它与命名本机查询类似)
hibernate 内部所做的是生成动态参数,在 IN 内部与传入列表中的元素数量相同。
因此,在上面的示例中,
如果列表有 2 个元素,则查询将如下所示
,如果列表有 3 个元素,则查询将如下所示
Tried in JPA2 with Hibernate as provider and it seems hibernate does support taking in a list for "IN" and it works. (At least for named queries and I believe it will be similar with named NATIVE queries)
What hibernate does internally is generate dynamic parameters, inside the IN same as the number of elements in the passed in list.
So in you example above
If list has 2 elements the query will look like
and if it has 3 elements it looks like
你应该这样做:
String userIds ="1,2,3,4,5";
列表<字符串> userIdList= Stream.of(userIds.split(",")).collect(Collectors.toList());
然后,在查询中传递 like 参数,如下所示:
@NamedNativeQuery(name=" User.findByUserIdList", query="从用户 u 中选择 u.user_id、u.dob、u.name、u.sex、u.address,其中 u.user_id in (?userIdList)")
you should do this:
String userIds ="1,2,3,4,5";
List<String> userIdList= Stream.of(userIds.split(",")).collect(Collectors.toList());
Then, passes like parameter inside your query, like this:
@NamedNativeQuery(name="User.findByUserIdList", query="select u.user_id, u.dob, u.name, u.sex, u.address from user u where u.user_id in (?userIdList)")
标准 JPA 是不可能的。 Hibernate 提供了专有方法
setParameterList()
,但它仅适用于 Hibernate 会话,并且在 JPA 的EntityManager
中不可用。我为 Hibernate 想出了以下解决方法,这并不理想,但几乎是标准的 JPA 代码,并且具有一些很好的属性。
对于初学者,您可以将命名的本机查询很好地分隔在
orm.xml
文件中:占位符用单引号括起来,因此它是有效的本机 JPA 查询。它在不设置参数列表的情况下运行,并且当在其周围设置其他匹配的颜色参数时仍然会返回正确的结果。
在 DAO 或存储库类中设置参数列表:
无需手动构造查询字符串。您可以像平常一样设置任何其他参数。
辅助方法:
基本上,我们从
orm.xml
读取查询字符串,手动设置参数列表,然后创建本机 JPA 查询。不幸的是,createNativeQuery().getResultList() 会返回一个非类型化查询和非类型化列表,即使我们向它传递了一个结果类。因此,@SuppressWarnings("unchecked")
。缺点:对于 Hibernate 以外的 JPA 提供者来说,在不执行查询的情况下展开查询可能会更复杂或不可能。例如,以下内容可能适用于 EclipseLink(未经测试,取自 我可以从 JPA 查询对象获取 SQL 字符串吗?):
另一种方法可能是将查询存储在文本文件中,并添加代码以从那里读取它。
It's not possible with standard JPA. Hibernate offers the proprietary method
setParameterList()
, but it only works with Hibernate sessions and is not available in JPA'sEntityManager
.I came up with the following workaround for Hibernate, which is not ideal but almost standard JPA code and has some nice properties to it.
For starters you can keep the named native query nicely separated in a
orm.xml
file:The placeholder is wrapped in single quotes, so it's a valid native JPA query. It runs without setting a parameter list and would still return correct results when other matching color parameters are set around it.
Set the parameter list in your DAO or repository class:
No manual construction of query strings. You can set any other parameter as you normally would.
Helper methods:
So basically we're reading a query string from
orm.xml
, manually set a parameter list and then create the native JPA query. Unfortunately,createNativeQuery().getResultList()
returns an untyped query and untyped list even though we passed a result class to it. Hence the@SuppressWarnings("unchecked")
.Downside: Unwrapping a query without executing it may be more complicated or impossible for JPA providers other than Hibernate. For example, the following might work for EclipseLink (untested, taken from Can I get the SQL string from a JPA query object?):
An alternative might be to store the query in a text file and add code to read it from there.
您可以传入列表作为参数,但是:
如果您创建
@NamedNativeQuery
并使用.createNamedQuery()
,则不使用命名参数,您使用了?1
(位置参数)。它从 1 开始,而不是 0。如果您使用
.createNativeQuery(String)
,则可以使用命名参数。You can pass in a list as a parameter, but:
if you create a
@NamedNativeQuery
and use.createNamedQuery()
, you don't use named param, you used?1
(positional parameter). It starts with 1, not 0.if you use
.createNativeQuery(String)
, you can use named param.您可以尝试使用 :userIdList 而不是 (?userIdList)
You can try this :userIdList instead of (?userIdList)
上述接受的答案不正确,导致我偏离轨道很多天!!
您只需要这样做
另请参阅此处提出相同建议的答案(我从其中之一中选择了上面的示例)
*请注意,这些参考文献是关于 jpql 查询的,尽管如此,集合的使用也适用于本机查询。
The above accepted answer is not correct and led me off track for many days !!
You just need to do
Also refer the answers here which suggest the same (I picked the above example from one of them)
*note that these references are about jpql queries, nevertheless the usage of collections is working with native queries too.
列表不是本机 SQL 查询的有效参数,因为它无法在 JDBC 中绑定。列表中的每个参数都需要有一个参数。
where u.user_id in (?id1, ?id2)
这是通过 JPQL 支持的,但不支持 SQL,因此您可以使用 JPQL 而不是本机查询。
某些 JPA 提供商可能支持此功能,因此您可能需要向提供商记录错误。
A list is not a valid parameter for a native SQL query, as it cannot be bound in JDBC. You need to have a parameter for each argument in the list.
where u.user_id in (?id1, ?id2)
This is supported through JPQL, but not SQL, so you could use JPQL instead of a native query.
Some JPA providers may support this, so you may want to log a bug with your provider.
可以很简单:
can be as simple as:
事实上,您可以将列表作为绑定参数传递给 JPA 本机查询,具体取决于您的数据库/提供程序/驱动程序/等。
例如,使用 Postgres 和 EclipseLink,可以执行以下操作(返回 true),演示多维数组以及如何获取双精度数组。 (对于其他类型,请执行
SELECT pg_type.* FROM pg_catalog.pg_type
;可能是带有_
的类型,但在使用之前将其剥离。)强制转换就在那里,因此文字数组是双精度数而不是数字。
更重要的是 - 我不知道如何或是否可以进行命名查询;我认为这可能取决于情况。但我认为以下内容适用于数组。
我没有与 OP 相同的模式,所以我没有完全检查这一点,但我认为它应该再次工作,至少在 Postgres & 上。 EclipseLink。
此外,该密钥位于: http://tonaconsulting.com/ postgres-and-multi-dimensions-arrays-in-jdbc/
Depending on your database/provider/driver/etc., you can, in fact, pass a list in as a bound parameter to a JPA native query.
For example, with Postgres and EclipseLink, the following works (returning true), demonstrating multidimensional arrays and how to get an array of double precision. (Do
SELECT pg_type.* FROM pg_catalog.pg_type
for other types; probably the ones with_
, but strip it off before using it.)The cast is there so the literal array is of doubles rather than numeric.
More to the point of the question - I don't know how or if you can do named queries; I think it depends, maybe. But I think following would work for the Array stuff.
I don't have the same schema as OP, so I haven't checked this exactly, but I think it should work - again, at least on Postgres & EclipseLink.
Also, the key was found in: http://tonaconsulting.com/postgres-and-multi-dimensions-arrays-in-jdbc/
使用 hibernate、JPA 2.1 和 deltaspike 数据,我可以将列表作为包含 IN 子句的查询中的参数传递。我的查询如下。
Using hibernate, JPA 2.1 and deltaspike data I could pass a list as parameter in query that contains IN clause. my query is below.
在jpa,它对我有用
In jpa, it worked for me
目前我将 JPA 2.1 与 Hibernate 一起使用,
我还在本机查询中使用 IN 条件。我的查询示例
我注意到,由于James描述的限制,不可能传递像“id_1,id_2,id_3”这样的字符串
但是当您使用 jpa 2.1 + hibernate 时,可以传递字符串值列表。对于我的情况,下一个代码是有效的:
currently I use JPA 2.1 with Hibernate
I also use IN condition with native query. Example of my query
I noticed that it's impossible to pass String like "id_1, id_2, id_3" because of limitations described by James
But when you use jpa 2.1 + hibernate it's possible to pass List of string values. For my case next code is valid: