O código a seguir pode ajudá-lo a obter a variável de
SYS_REFCURSOR do bloco anônimo Oracle. Devemos nos concentrar em alguns detalhes importantes:
- Classe
groovy.sql.Sqlnão temOutParametercorrespondente e fazemos manualmente comoCURSOR_PARAMETERe passe parasql.callmétodo - Considere que o bloco começa com
{call DECLAREe termina comEND }sem ponto e vírgula após END. Caso contrário, podemos obter umaSQLExceptionpouco reconhecível na cara. - Os pontos de interrogação
?dentro dosqlStringsão locais para ligações de parâmetros. As ligações são feitas na ordem natural tomando valores deparametersList.- Neste exemplo, temos a única ligação, então o
?vincula com CURSOR_PARAMETER tratando o valor comoOUTparâmetro do tipo passado;
- Neste exemplo, temos a única ligação, então o
- Há apenas uma entrada em encerramento após
sql.calleResultSet rsfornecer linhas de cursormy_curdeclarado em bloco anônimo. - Podemos simplificar
sqlStringusando uma função que retornaSYS_REFCURSORao invés de um procedimento comOUTparâmetro. Então poderia ficar assim"{call BEGIN ? := MY_FUNC(); END}"ou mesmo"{? = call MY_FUNC()}"
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;
BEGIN
STORED_PROCEDURE_NAME(my_cur);
? := my_cur;
END
}
""";
// the order of elements matches the order of bindings
def parametersList = [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")
}
};
P.S. E obrigado pela pergunta.