JDBC работает с типами Oracle RECORD, TABLE и OBJECT - PullRequest
0 голосов
/ 14 февраля 2019

Как получить типы коллекций Oracle в Jdbc.Я много искал, но не нашел ответа.Ниже моя хранимая процедура работает нормально:

create or replace PACKAGE SHOW_SUP AS
  TYPE SUP_COF_REC IS RECORD (
    SUPPLIER_NAME VARCHAR2(32),
    COFFEE_NAME VARCHAR(32)
  );
TYPE_RECORD SUP_COF_REC;
TYPE SUP_COF_TAB IS TABLE OF TYPE_RECORD%TYPE;
PROCEDURE SHOW_SUPPLIERS (SUP_COF_DET OUT SUP_COF_TAB);
END SHOW_SUP;

create or replace PACKAGE BODY SHOW_SUP AS
PROCEDURE SHOW_SUPPLIERS (SUP_COF_DET OUT SUP_COF_TAB) AS
 STR VARCHAR2(2000);
 BEGIN
  STR:='SELECT SUP_NAME, COF_NAME FROM SUPPLIERS S, COFFEES C WHERE
        S.SUP_ID=C.SUP_ID ORDER BY SUP_NAME';
  EXECUTE IMMEDIATE(STR) BULK COLLECT INTO SUP_COF_DET;
 END SHOW_SUPPLIERS;
END SHOW_SUP;

Ниже приведен код Java для запуска хранимой процедуры:

String sql="{call SHOW_SUP.SHOW_SUPPLIERS(?)}";
CallableStatement cs = con.prepareCall(sql);
cs.registerOutParameter(1, java.sql.Types.ARRAY, "SUP_COF_TAB"); //LINE 3
//cs.registerOutParameter(1, java.sql.Types.ARRAY, "SHOW_SUP.SUP_COF_TAB"); //LINE 4
cs.execute();

Ошибка, которую я получаю на линии 3/4:

SQLException : invalid name pattern: SCOTT.SUP_COF_TAB
java.sql.SQLException: invalid name pattern: SCOTT.SUP_COF_TAB
    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)
    at oracle.jdbc.driver.DatabaseError.throwSqlException(DatabaseError.java:263)
    at oracle.jdbc.driver.DatabaseError.throwSqlException(DatabaseError.java:271)
    at oracle.jdbc.oracore.OracleTypeADT.initMetadata(OracleTypeADT.java:543)
    at oracle.jdbc.oracore.OracleTypeADT.init(OracleTypeADT.java:462)
    at oracle.sql.ArrayDescriptor.initPickler(ArrayDescriptor.java:1678)
    at oracle.sql.ArrayDescriptor.<init>(ArrayDescriptor.java:291)
    at oracle.sql.ArrayDescriptor.createDescriptor(ArrayDescriptor.java:206)
    at oracle.sql.ArrayDescriptor.createDescriptor(ArrayDescriptor.java:175)
    at oracle.sql.ArrayDescriptor.createDescriptor(ArrayDescriptor.java:158)
    at oracle.sql.ArrayDescriptor.createDescriptor(ArrayDescriptor.java:125)
    at oracle.jdbc.driver.NamedTypeAccessor.otypeFromName(NamedTypeAccessor.java:84)
    at oracle.jdbc.driver.TypeAccessor.initMetadata(TypeAccessor.java:89)
    at oracle.jdbc.driver.T4CCallableStatement.allocateAccessor(T4CCallableStatement.java:629)
    at oracle.jdbc.driver.OracleCallableStatement.registerOutParameterInternal(OracleCallableStatement.java:166)
    at oracle.jdbc.driver.OracleCallableStatement.registerOutParameter(OracleCallableStatement.java:207)
    at oracle.jdbc.driver.OracleCallableStatementWrapper.registerOutParameter(OracleCallableStatementWrapper.java:1584)
    at com.jdbc.callstatement.CallableStatementSample.main(CallableStatementSample.java:48)

Я также пытался запустить новую хранимую процедуру без типа RECORD , но существует та же ошибка:

create or replace PACKAGE SHOW_COF AS
TYPE COF_TAB IS TABLE OF COFFEES%ROWTYPE INDEX BY PLS_INTEGER;
PROCEDURE SHOW_COFFEES (COF_DET OUT COF_TAB);
END SHOW_COF;

Я где-то обнаружил, что дескриптор массива или дескриптор структуры можно использовать, нокак использовать его в registerOutParameter, не любой документ / сайт не имеет объяснения.Я застрял здесь.

Ответы [ 2 ]

0 голосов
/ 27 февраля 2019

Используйте полное имя структурированного типа SQL, как показано ниже:

cs.registerOutParameter(1, java.sql.Types.ARRAY, "SHOW_SUP.SUP_COF_TAB"); //LINE 3.

Чтобы прочитать параметр out, выполните следующие действия в своем коде Java:

// Get the out parameter as java.sql.Array
java.sql.Array ntab = (java.sql.Array) cStmt.getObject(1);

// Now get the rows from the nested table
Object[] rows = (Object[])ntab.getArray();

// For each row
for (Object row: rows) {
    // Get the columns
    Object[] cols = ((Struct) row).getAttributes();

    String supplierName = cols[0].toString();
    String coffeeName = cols[1].toString();
}
0 голосов
/ 15 февраля 2019

Это невозможно.Вы можете работать только с объектом уровня SQL в JDBC (UDT, коллекции).Но в вашем случае вы можете использовать sys_refcursors.В Java sys_refcursor представлен обычным набором результатов.

Понятие ниже:

ORACE:

 PROCEDURE SHOW_SUPPLIERS (SUP_COF_DET OUT SYS_REFCURSOR);
..

    PROCEDURE SHOW_SUPPLIERS (SUP_COF_DET OUT SYS_REFCURSOR) AS
     c_cursor SYS_REFCURSOR
     BEGIN
      open c_cursor  for 'SELECT SUP_NAME, COF_NAME FROM SUPPLIERS S, COFFEES C WHERE
            S.SUP_ID=C.SUP_ID ORDER BY SUP_NAME';
      SUP_COF_DET := c_cursor 
     END SHOW_SUPPLIERS;

Java:

String sql="{call SHOW_SUP.SHOW_SUPPLIERS(?)}";
CallableStatement cs = con.prepareCall(sql);
cs.registerOutParameter(1, OracleTypes.CURSOR);
cs.executeUpdate();
ResultSet rs = (ResultSet) cs.getObject(1);
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...