org.hibernate.QueryException: Space is not allowed after parameter prefix ':'解决

本文介绍了一个在使用Hibernate框架时遇到的参数定义错误导致的异常情况。具体表现为在SQL语句中参数占位符':n'前后误加空格,导致程序抛出org.hibernate.QueryException异常。通过调整SQL语句中参数的格式,成功解决了该问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

环境

Eclipse Indigo + Hibernate3.2 + Oracle 11g R2 + SQLPlus

问题

运行Hibernate程序,报如下异常

org.hibernate.QueryException: Space is not allowed after parameter prefix ':' [select id from contactor where name =: n]
	at org.hibernate.engine.query.ParameterParser.parse(ParameterParser.java:92)
	at org.hibernate.engine.query.ParamLocationRecognizer.parseLocations(ParamLocationRecognizer.java:75)
	at org.hibernate.engine.query.QueryPlanCache.buildNativeSQLParameterMetadata(QueryPlanCache.java:149)
	at org.hibernate.engine.query.QueryPlanCache.getSQLParameterMetadata(QueryPlanCache.java:79)
	at org.hibernate.impl.AbstractSessionImpl.createSQLQuery(AbstractSessionImpl.java:146)
	at org.hibernate.impl.SessionImpl.createSQLQuery(SessionImpl.java:1657)
	at com.addbook.dao.impl.ContactorDaoImpl.getContactorId(ContactorDaoImpl.java:90)
	at com.addbook.service.impl.ContactorServiceImpl.getContactorId(ContactorServiceImpl.java:58)
	at com.addbook.dao.impl.TestContactor.testGetContactorId(TestContactor.java:71)
	at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
	at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
	at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
	at java.lang.reflect.Method.invoke(Method.java:597)
	at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:44)
	at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:15)
	at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:41)
	at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:20)
	at org.junit.runners.BlockJUnit4ClassRunner.runNotIgnored(BlockJUnit4ClassRunner.java:79)
	at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:71)
	at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:49)
	at org.junit.runners.ParentRunner$3.run(ParentRunner.java:193)
	at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:52)
	at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:191)
	at org.junit.runners.ParentRunner.access$000(ParentRunner.java:42)
	at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:184)
	at org.junit.runners.ParentRunner.run(ParentRunner.java:236)
	at org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:50)
	at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38)
	at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:467)
	at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:683)
	at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:390)
	at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:197)


解决

引用占位符的前后不允许有空格。

sql = "select id from contactor where name=:n";


`java.sql.SQLException: ResultSet operation not allowed after ResultSet closed` 这是一个常见的 SQL 异常错误提示,表示尝试对已经关闭的 `ResultSet` 对象进行操作。 ### 原因分析 1. **资源未正确管理**:通常在数据库查询完成后,`ResultSet`, `Statement` 和 `Connection` 都需要手动关闭。如果过早地关闭了 `ResultSet` 或其关联的 `Statement`、`Connection`,那么后续对该 `ResultSet` 的访问就会抛出这个异常。 2. **游标的生命周期结束**:SQL 查询结果是由数据库服务器返回的一个“指针”或“游标”,当该游标被显式或隐式关闭后,再次对其进行读取会失败。 3. **连接池的行为影响**:如果你使用的是一种数据库连接池技术,在某些框架下可能会自动回收和关闭底层的对象,而此时程序仍试图继续处理已关闭的结果集。 ### 解决方案 为了避免上述问题发生,请注意以下几点: - 确保所有相关的对象 (`ResultSet`, `Statement`, `PreparedStatement`) 按正确的顺序关闭,并尽量靠近它们作用域末端的位置释放; - 尽量将数据从 `ResultSet` 中提取出来并存储到本地变量或其他结构(如 List<Map<String, Object>>),然后再断开与数据库之间的物理链接; ```java try (Connection conn = DriverManager.getConnection(DB_URL); Statement stmt = conn.createStatement(); ResultSet rs = stmt.executeQuery("SELECT * FROM table")) { while(rs.next()){ // Process each row... } } catch(SQLException e){ System.err.println(e.getMessage()); } ``` 通过使用 Java7+ 推荐的 try-with-resources 自动化机制可以有效避免内存泄漏及类似本题这样的运行期错误情况出现。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值