我的无效字符在哪里 (ORA-00911)

Posted

技术标签:

【中文标题】我的无效字符在哪里 (ORA-00911)【英文标题】:Where's my invalid character (ORA-00911) 【发布时间】:2012-05-30 11:58:51 【问题描述】:

我正在尝试将CLOBs 插入数据库(请参阅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)的主要内容,如果未能解决你的问题,请参考以下文章

ORA-00911: C# 中的无效字符

了解神秘的 Oracle JDBC 错误 - ORA-00911:无效字符

ORA-00911: 无效字符 00911. 00000 - Oracle 中出现“无效字符”异常错误

Mybatis 批量更新 ORA-00911: 无效字符的错误

java.sql.SQLException: ORA-00911: 无效字符

用作子查询时的分层查询给出:ORA-00911:无效字符