我的无效字符在哪里 (ORA-00911)
Posted
技术标签:
【中文标题】我的无效字符在哪里 (ORA-00911)【英文标题】:Where's my invalid character (ORA-00911) 【发布时间】:2012-05-30 11:58:51 【问题描述】:我正在尝试将CLOB
s 插入数据库(请参阅related question)。我无法完全弄清楚出了什么问题。我有一个要插入到表中的大约 85 个 clob 的列表。即使只插入第一个 clob,我也会得到ORA-00911: invalid character
。我无法弄清楚如何在执行之前将语句从PreparedStatement
中取出,所以我不能 100% 确定它是正确的,但如果我做对了,那么它应该看起来完全像这样:
insert all
into domo_queries values ('select
substr(to_char(max_data),1,4) as year,
substr(to_char(max_data),5,6) as month,
max_data
from dss_fin_user.acq_dashboard_src_load_success
where source = ''CHQ PeopleSoft FS''')
select * from dual;
最终,这个insert all
语句会有很多into
,这就是为什么我不做一个常规的insert
语句。我没有在其中看到无效字符,是吗? (哦,上面的代码在我的 sql 开发人员工具中运行时运行良好。)如果我删除 PreparedStatement
中的分号,它会引发 ORA-00933: SQL command not properly ended
错误。
无论如何,这是我执行查询的代码(以及上面示例的变量值)。
public ResultSet executeQuery(String connection, String query, QueryParameter... params) throws DataException, SQLException
// query at this point = "insert all
//into domo_queries values (?)
//select * from dual;"
Connection conn = ConnectionPool.getInstance().get(connection);
PreparedStatement pstmt = conn.prepareStatement(query);
for (int i = 1; i <= params.length; i++)
QueryParameter param = params[i - 1];
switch (param.getType()) //The type in the example is QueryParameter.CLOB
case QueryParameter.CLOB:
Clob clob = CLOB.createTemporary(conn, false, oracle.sql.CLOB.DURATION_SESSION);
clob.setString(i, "'" + param.getValue() + "'");
//the value of param.getValue() at this point is:
/*
* select
* substr(to_char(max_data),1,4) as year,
* substr(to_char(max_data),5,6) as month,
* max_data
* from dss_fin_user.acq_dashboard_src_load_success
* where source = ''CHQ PeopleSoft FS''
*/
pstmt.setClob(i, clob);
break;
case QueryParameter.STRING:
pstmt.setString(i, "'" + param.getValue() + "'");
break;
ResultSet rs = pstmt.executeQuery(); //Obviously, this is where the error is thrown
conn.commit();
ConnectionPool.getInstance().release(conn);
return rs;
我错过了什么?
【问题讨论】:
你能把 PreparedStatement 看成一个字符串来看看它包含什么吗? 我在帖子中提到过。我不知道该怎么做。我在互联网上看到的只是由于某种原因它非常复杂...... 一些 JDBC 驱动程序允许 toString() 向您显示来自 PreparedStatement 的查询,而有些则不允许。不确定甲骨文。 @DanArmstrong,不幸的是,这里不是这种情况。我在toString()
打印输出上得到oracle.jdbc.driver.OraclePreparedStatementWrapper@8870a2
。
Oracle 是否支持查询日志记录。如果是这样,那么您可以在查询到达服务器时看到它。
【参考方案1】:
如果您完全按照向我们展示的方式使用字符串文字,那么问题在于末尾的 ;
字符。您可能不会在 JDBC 调用的查询字符串中包含它。
由于您只插入单行,因此即使插入多行,常规 INSERT
也应该没问题。无论如何,使用批处理语句可能更有效。不需要INSERT ALL
。此外,您不需要临时 clob 和所有这些。你可以把你的方法简化成这样(假设我的参数是正确的):
String query1 = "select substr(to_char(max_data),1,4) as year, " +
"substr(to_char(max_data),5,6) as month, max_data " +
"from dss_fin_user.acq_dashboard_src_load_success " +
"where source = 'CHQ PeopleSoft FS'";
String query2 = ".....";
String sql = "insert into domo_queries (clob_column) values (?)";
PreparedStatement pstmt = con.prepareStatement(sql);
StringReader reader = new StringReader(query1);
pstmt.setCharacterStream(1, reader, query1.length());
pstmt.addBatch();
reader = new StringReader(query2);
pstmt.setCharacterStream(1, reader, query2.length());
pstmt.addBatch();
pstmt.executeBatch();
con.commit();
【讨论】:
补充一点,很多人认为;
是Oracle SQL中的语句终止符。它不是。客户端(例如 SQLPlus)使用语句末尾的 ;
来告知语句在哪里结束,然后发送语句而不是“;”到 Oracle 服务器。在 SQLPLus 中,使用默认值尝试 select * from dual; --semicolon to terminate
。对于命令的其余部分,您将收到 2
提示,因为 SQL*Plus 仅在 ;
是一行中的最后一个字符时才使用它,但 select * from dual --weird that this works;
;
似乎是注释的一部分,但仍然“终止”声明。
@ShannonSeverance:;
被定义为 SQL 标准的语句终止字符。但是为了增加对此的混淆:SQL Server(不是客户端!)要求一些语句在最后通过 JDBC 发送;
。显然它有时也需要在声明的前面加上;
。
我试图表明我在讨论 Oracle 的 SQL 实现,而不是标准 SQL。 SQL Server 很古怪,有时需要;
来分隔语句,但一直不需要;
。
上面的注释应该是:PL/SQL 确实使用;
作为语句终止符,但它是一种语言,与Oracle 的SQL 实现不同。我>
只是一个;杀了我【参考方案2】:
在我的脑海中,你能尝试使用 'q' 运算符作为字符串文字
类似
insert all
into domo_queries values (q'[select
substr(to_char(max_data),1,4) as year,
substr(to_char(max_data),5,6) as month,
max_data
from dss_fin_user.acq_dashboard_src_load_success
where source = 'CHQ PeopleSoft FS']')
select * from dual;
请注意,谓词的单引号没有转义,字符串位于 q'[...]' 之间。
【讨论】:
太棒了,我忘了q'[]'
。感谢您的提示!【参考方案3】:
其中一个原因可能是任何一个表列的名称中都有一个下划线(_)。这被 JDBC 视为无效字符。通过 ALTER 命令重命名列并更改您的代码 SQL,这将修复。
【讨论】:
以上是关于我的无效字符在哪里 (ORA-00911)的主要内容,如果未能解决你的问题,请参考以下文章
了解神秘的 Oracle JDBC 错误 - ORA-00911:无效字符
ORA-00911: 无效字符 00911. 00000 - Oracle 中出现“无效字符”异常错误
Mybatis 批量更新 ORA-00911: 无效字符的错误