使用位置参数的Hibernate本地查询

5

我写了本地SQL查询,而不是使用HQL,并遇到了问题。

位置超出已声明的序数参数。请记住,序数参数从1开始计数!位置:1

<sql-query name="GET_ARRAY_MAX_POINT_QUESTION">
    <![CDATA[
        select TEST.TEST_ID as testId, TEST.VERSION_ID as versionId,
        PASSED_TEST.RESULT as userResult,
        PASSED_TEST.TIME_COMPLITED as timeComplited,
        sum(COMPLEXITY) as maxTestResult from QUESTION
        JOIN TEST_QUESTION ON QUESTION.QUESTION_ID = TEST_QUESTION.QUESTION_ID
        JOIN TEST ON TEST.TEST_ID=TEST_QUESTION.TEST_ID
        JOIN PASSED_TEST ON TEST.TEST_ID=PASSED_TEST.TEST_ID
        AND TEST.VERSION_ID=PASSED_TEST.VERSION_ID
        WHERE TEST.SUBJECT_ID = ?
        AND PASSED_TEST.USER_ID = ?
        GROUP BY PASSED_TEST.TEST_EVENT_ID
        ]]>
    </sql-query>

和DAO
return session
                .createSQLQuery(GET_ARRAY_MAX_POINT_QUESTION_NAME_QUERY)
                .addScalar(TEST_ID_RESULT_PARAM, StandardBasicTypes.LONG)
                .addScalar(VERSION_ID_RESULT_PARAM, StandardBasicTypes.LONG)
                .addScalar(USER_RESULT_PARAM, StandardBasicTypes.DOUBLE)
                .addScalar(MAX_TEST_RESULT_PARAM, StandardBasicTypes.DOUBLE)
                .addScalar(TIME_COMPLITED_RESULT_PARAM, StandardBasicTypes.DATE)
                .setParameter(0, subjectId)
                .setParameter(1, userId)
                .setResultTransformer(
                        Transformers.aliasToBean(PassedTestStatistic.class))
                .list();

我看到过这样的话://JPA规范。只有位置参数绑定可用于本地查询。而Hibernate使用0作为第一个索引。

堆栈轨迹

Caused by: org.hibernate.QueryParameterException: Position beyond number of declared ordinal parameters. Remember that ordinal parameters are 1-based! Position: 1
    at org.hibernate.engine.query.spi.ParameterMetadata.getOrdinalParameterDescriptor(ParameterMetadata.java:80)
    at org.hibernate.engine.query.spi.ParameterMetadata.getOrdinalParameterExpectedType(ParameterMetadata.java:86)
    at org.hibernate.internal.AbstractQueryImpl.determineType(AbstractQueryImpl.java:444)
    at org.hibernate.internal.AbstractQueryImpl.setParameter(AbstractQueryImpl.java:416)
    at by.bsuir.testapp.database.hibernate.PassedTestHibernateDAO.getDataForPassedTestStatisticGraph(PassedTestHibernateDAO.java:73)
    at by.bsuir.testapp.service.PassedTestServiceImpl.getDataForPassedTestStatisticGraph(PassedTestServiceImpl.java:58)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:601)
    at org.springframework.aop.support.AopUtils.invokeJoinpointUsingReflection(AopUtils.java:319)
    at org.springframework.aop.framework.ReflectiveMethodInvocation.invokeJoinpoint(ReflectiveMethodInvocation.java:183)
    at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:150)
    at org.springframework.transaction.interceptor.TransactionInterceptor.invoke(TransactionInterceptor.java:110)
    at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:172)
    at org.springframework.aop.framework.JdkDynamicAopProxy.invoke(JdkDynamicAopProxy.java:202)
    at $Proxy28.getDataForPassedTestStatisticGraph(Unknown Source)
    at by.bsuir.testapp.controller.StatisticPassedTest.createLinearModel(StatisticPassedTest.java:61)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:601)
    at org.apache.el.parser.AstValue.invoke(AstValue.java:262)
    at org.apache.el.MethodExpressionImpl.invoke(MethodExpressionImpl.java:278)

//更新

有趣的是,当我设置查询数字值时

WHERE TEST.SUBJECT_ID = 1
        AND PASSED_TEST.USER_ID = 1

我明白了。

Caused by: com.mysql.jdbc.exceptions.jdbc4.MySQLSyntaxErrorException: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near 'GET_ARRAY_MAX_POINT_QUESTION' at line 1 

但是在MySQL中我得到了成功的结果。

我该如何解决这个问题?


那问题是什么? - JB Nizet
@JB Nizet 位置超出了声明的序数参数数目。请记住,序数参数是基于1的!位置:1 - Ray
请粘贴完整的堆栈跟踪。 - JB Nizet
Hibernate 的哪个版本? - schtever
@Ray 你尝试过将 createSQLQuery 更改为 getNamedQuery 吗?这是一些答案中建议的,我认为这很重要。 - schtever
@schtever,这是SQL查询而不是HQL,有关系吗? - Ray
5个回答

7

You call

session.createSQLQuery 

但我认为你需要调用

session.getNamedQuery 

在使用命名查询时。

2

请确认

.setParameter(1, subjectId)
.setParameter(2, userId)

异常跟踪显示序数参数是基于1的!


异常似乎是误导性的。实际上它是基于0的。 - Rahul

1

我尝试使用位置参数从0开始,也从1开始,还使用了命名参数。 - Ray

1

查看您的源代码

createSQLQuery(GET_ARRAY_MAX_POINT_QUESTION_NAME_QUERY)

你必须有一个指向包含本地查询实际名称的字符串的变量,如下所示:

GET_ARRAY_MAX_POINT_QUESTION_NAME_QUERY = "GET_ARRAY_MAX_POINT_QUESTION"

在这里您收到的错误信息是:
Caused by: com.mysql.jdbc.exceptions.jdbc4.MySQLSyntaxErrorException: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near 'GET_ARRAY_MAX_POINT_QUESTION' at line 1

这意味着 createSqlQuery 没有按名称找到您的本地查询,而是尝试将名称“GET_ARRAY_MAX_POINT_QUESTION”作为原始 SQL 执行,就像您输入 s.createSqlQuery("Select * from...") 一样。如果您查看文档 此处,您会发现即使是本地查询,要查找命名查询,仍然必须使用 getNamedQuery。


0
对于那些谷歌错误消息的人: 当我没有一个类的映射时,遇到了这个错误。映射解决了这个问题。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接