sql - 如何使用 Groovy 从 Oracle 获取光标?

标签 sql oracle jdbc groovy mule

我正在使用 Groovy Mule ESB 中的脚本从 Oracle 获取输出参数存储过程(包括游标)并获得异常。

最小的例子:

import groovy.sql.Sql
import oracle.jdbc.pool.OracleDataSource
import oracle.jdbc.driver.OracleTypes

def ds = new OracleDataSource()
// setting data source parameters here

def sql = new Sql(ds)
def data = []

sql.call("""declare
result_table sys_refcursor;
begin

open result_table for select 1 as a from dual;

insert into CURSOR_TEST (ID) values (1);
commit;

${Sql.resultSet OracleTypes.CURSOR} := result_table;

insert into CURSOR_TEST (ID) values (2);
commit;

end;
"""
){ table ->

  throw new RuntimeException("Never getting this exception.")

  table.eachRow {
    data << it.toRowResult()
  }
}

sql.close()

return data

错误:


Message               : java.sql.SQLException: Closed Statement (javax.script.ScriptException)
Code                  : MULE_ERROR--2
--------------------------------------------------------------------------------
Exception stack is:
1. Closed Statement(SQL Code: 17009, SQL State: + 99999) (java.sql.SQLException)
  oracle.jdbc.driver.SQLStateMapping:70 (null)
2. java.sql.SQLException: Closed Statement (javax.script.ScriptException)
  org.codehaus.groovy.jsr223.GroovyScriptEngineImpl:323 (http://java.sun.com/j2ee/sdk_1.3/techdocs/api/javax/script/ScriptException.html)
3. java.sql.SQLException: Closed Statement (javax.script.ScriptException)

(org.mule.api.transformer.TransformerException) org.mule.module.scripting.transformer.ScriptTransformer:39 (http://www.mulesoft.org/docs/site/current3/apidocs/org/mule/api/transformer/TransformerException.html) -------------------------------------------------------------------------------- Root Exception stack trace: java.sql.SQLException: Closed Statement at oracle.jdbc.driver.SQLStateMapping.newSQLException(SQLStateMapping.java:70) at oracle.jdbc.driver.DatabaseError.newSQLException(DatabaseError.java:133) at oracle.jdbc.driver.DatabaseError.throwSqlException(DatabaseError.java:199) + 3 more (set debug level logging or '-Dmule.verbose.exceptions=true' for everything) ********************************************************************************


Select来自 CURSOR_TEST返回 12 .

Oracle 服务器版本:Oracle Database 11g Enterprise Edition Release 11.2.0.3.0 - 64bit Production .

骡版:3.5.0 .

我正在使用 jdbc\lib\ojdbc6.jar来自 oracle 客户端版本 11.1.0.7.0 .

我究竟做错了什么?

最佳答案

以下代码可以帮助您获得SYS_REFCURSOR的变量来自 Oracle 匿名 block 。

我们应该关注几个关键细节:

  • 类(class) groovy.sql.Sql没有对应的OutParameter我们手动将其设置为 CURSOR_PARAMETER并将其传递给 sql.call方法
  • 考虑该 block 以 {call DECLARE 开头并以 END } 结尾END 后不加分号。否则我们会得到一个难以识别的SQLException在脸上。
  • 问号?里面sqlString是参数绑定(bind)的地方。绑定(bind)是按自然顺序制作的。在这个例子中:
  • 第一个?parametersList 中的第一个元素绑定(bind): "abc" , 将值视为 IN范围 ;
  • 第二个?与 CURSOR_PARAMETER 绑定(bind),将值视为 OUT传递类型的参数;
  • sql.call之后只有一个进入关闭和 ResultSet rs提供游标行my_cur在匿名 block 中声明。

  • import groovy.sql.OutParameter
    import groovy.sql.Sql
    import oracle.jdbc.OracleTypes
    
    import java.sql.ResultSet
    
    def driver = 'oracle.jdbc.driver.OracleDriver'
    def sql = Sql.newInstance('jdbc:oracle:thin:@MY-SERVER:1521:XXX', 'usr', 'psw', driver)
    
    // special OutParameter for cursor type
    OutParameter CURSOR_PARAMETER = new OutParameter() {
        public int getType() {
            return OracleTypes.CURSOR;
        }
    };
    
    // look at some ceremonial wrappers around anonymous block
    String sqlString = """{call
        DECLARE
          my_cur SYS_REFCURSOR;
          x VARCHAR2(32767) := ?;
        BEGIN
    
            OPEN my_cur
            FOR
            SELECT x || level AS my_column FROM dual CONNECT BY level < 10;
    
            ? := my_cur;
        END
    }
    """;
    
    // the order of elements matches the order of bindings
    def parametersList = ["abc", CURSOR_PARAMETER];
    
    
    // rs contains the result set of cursor my_cur
    sql.call(sqlString, parametersList) { ResultSet rs ->
      while (rs.next()) {
          println rs.getString("my_column")
      }
    };
    

    关于sql - 如何使用 Groovy 从 Oracle 获取光标?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20242512/

    相关文章:

    oracle - 行数与 oracle 中的类似查询不匹配

    maven - 为什么 Spring Boot 应用程序需要 hikariCP?

    java - Oracle JDBC 选择 WHERE 返回 0

    regex - 从大约 5 个单词中以任意顺序匹配至少 3 个单词

    sql - 如何使用 SQL 创建开始日期和结束日期?

    c# - DataTable 到 CSV 日期格式

    mysql - 错误代码 1136 : INTERT INTO COMMAND

    Oracle 从两个表中选择数据,如果一个为空返回 null

    java - C3p0 - UnreturnedConnectionTimeout 在被 C3p0 抛出时不会破坏流控制

    mysql - 应用触发器后受影响的行数