3

JPA/Hibernate 查询中允许的连接数是否有限制?

由于 Hibernate不会自动加入,因此我必须在 JPA/Hibernate 查询中明确指定联接。例如,人有地址,地址有状态。以下查询检索地址和状态已满载的人员:

select p, a, s from person p left join p.address a left join a.state s where ...

随着我不断添加连接,我最终(在 12-13 个左连接之后)达到了 Hibernate 生成无效 SQL 的限制:

Caused by: java.sql.SQLException: Column 'something69_2_' not found. 

我确实为我的数据库实现 MySQL 设置了 Hibernate 的方言:

<property name="hibernate.dialect">org.hibernate.dialect.MySQLInnoDBDialect</property>

Hibernate 在单个查询中可以处理的连接数是否有限制?

编辑1:日志文件中包含以下内容:

could not read column value from result set: something69_2_; Column 'something69_2_' not found.

但是,something69_2_不会出现在 SQL 查询中。这就像 Hibernate 生成了一个 SQL 查询并期望something69_2_在结果中,但事实并非如此。

编辑 2:类似问题记录为未修复的Hibernate 错误 HHH-3035

编辑 3:这是一个记录在案的Hibernate 错误 HHH-3636,已修复但尚未包含在任何版本中。

编辑 4:我构建了 hibernate-core 3.3.2-SNAPSHOT,其中包括错误修复 HHH-3636,但它没有解决这个问题。

编辑 5:错误行为似乎是由LEFT JOIN FETCHManyToMany 或 OneToMany 关系上的多个触发的。一个会起作用,两个或三个会导致错误。

编辑 6:这是堆栈跟踪:

javax.persistence.PersistenceException: org.hibernate.exception.SQLGrammarException: could not execute query
        at org.hibernate.ejb.AbstractEntityManagerImpl.throwPersistenceException(AbstractEntityManagerImpl.java:629)
        at org.hibernate.ejb.QueryImpl.getResultList(QueryImpl.java:73)
Caused by: org.hibernate.exception.SQLGrammarException: could not execute query
        at org.hibernate.exception.SQLStateConverter.convert(SQLStateConverter.java:67)
        at org.hibernate.exception.JDBCExceptionHelper.convert(JDBCExceptionHelper.java:43)
        at org.hibernate.loader.Loader.doList(Loader.java:2214)
        at org.hibernate.loader.Loader.listIgnoreQueryCache(Loader.java:2095)
        at org.hibernate.loader.Loader.list(Loader.java:2090)
        at org.hibernate.loader.hql.QueryLoader.list(QueryLoader.java:388)
        at org.hibernate.hql.ast.QueryTranslatorImpl.list(QueryTranslatorImpl.java:338)
        at org.hibernate.engine.query.HQLQueryPlan.performList(HQLQueryPlan.java:172)
        at org.hibernate.impl.SessionImpl.list(SessionImpl.java:1121)
        at org.hibernate.impl.QueryImpl.list(QueryImpl.java:79)
        at org.hibernate.ejb.QueryImpl.getResultList(QueryImpl.java:64)
        ... 69 more
Caused by: java.sql.SQLException: Column 'something69_2_' not found.
        at com.mysql.jdbc.SQLError.createSQLException(SQLError.java:1055)
        at com.mysql.jdbc.SQLError.createSQLException(SQLError.java:956)
        at com.mysql.jdbc.SQLError.createSQLException(SQLError.java:926)
        at com.mysql.jdbc.ResultSetImpl.findColumn(ResultSetImpl.java:1136)
        at com.mysql.jdbc.ResultSetImpl.getInt(ResultSetImpl.java:2777)
        at org.hibernate.type.IntegerType.get(IntegerType.java:28)
        at org.hibernate.type.NullableType.nullSafeGet(NullableType.java:113)
        at org.hibernate.type.NullableType.nullSafeGet(NullableType.java:102)
        at org.hibernate.loader.Loader.getKeyFromResultSet(Loader.java:1088)
        at org.hibernate.loader.Loader.getRowFromResultSet(Loader.java:553)
        at org.hibernate.loader.Loader.doQuery(Loader.java:689)
        at org.hibernate.loader.Loader.doQueryAndInitializeNonLazyCollections(Loader.java:224)
        at org.hibernate.loader.Loader.doList(Loader.java:2211)
        ... 77 more

编辑 7:所有这些连接的原因是为了避免 Hibernate 执行 n+1 查询,请参阅Hibernate FAQ 关于如何在运行 Hibernate 查询时避免 n+1 SQL SELECT 查询?

4

5 回答 5

2

我曾经使用 Hibernate 达到 MySQL 5.0 61 表的限制:

ERROR 1116 (HY000):表太多;MySQL 在一个连接中只能使用 61 个表
于 2009-03-13T22:14:01.550 回答
2

问题是您为什么首先尝试进行如此复杂的查询?

您是否考虑过不同的方法?关于提高性能的文档提出了一些建议。

于 2009-03-15T22:04:12.533 回答
1

Hibernate 代码中没有限制连接数量的内容。这可能是方言中的错误,也可能是数据库引擎的限制。但是我的钱花在了一个与加入数量无关的错误上!您是否尝试过在交互式查询会话中直接运行 SQL?

于 2009-01-21T22:26:57.900 回答
0

您是否尝试过使用实际使用的 jdbc 驱动程序执行?可能是 jdbc 驱动的问题。

虽然从列的名称来看,它正在寻找,我猜名称的一些修剪/构造出错了。绝对看起来更像是一个错误而不是预期的限制。

于 2009-03-15T16:36:55.270 回答
0

您是否正在为所有内部连接起别名?并使用这些别名?当您尝试在子句中使用隐式别名来执行以下操作时,我已经看到 Hibernate 的一些非常奇怪的行为SELECT(极其简化的示例):

select p.address.state from person p

但是,如果您明确声明所有别名,它就可以正常工作。像这样:

select s from person p join p.address a join a.state s

...甚至是这个:

select s from person p join p.address.state s
于 2012-12-05T15:03:49.433 回答