Как вернуть набор результатов / курсор из анонимного блока Oracle PL / SQL, который выполняет динамический SQL? - PullRequest
23 голосов
/ 28 января 2010

У меня есть эта таблица:

ALLITEMS
---------------
ItemId  | Areas
---------------
1       | EAST
2       | EAST
3       | SOUTH
4       | WEST

DDL:

drop table allitems;

Create Table Allitems(ItemId Int,areas Varchar2(20));
Insert Into Allitems(Itemid,Areas) Values(1,'east');
Insert Into Allitems(ItemId,areas) Values(2,'east');
insert into allitems(ItemId,areas) values(3,'south');
insert into allitems(ItemId,areas) values(4,'east');

В MSSQL, чтобы получить курсор из динамического SQL, я могу сделать:

DECLARE @v_sqlStatement VARCHAR(2000);
SET @v_Sqlstatement = 'SELECT * FROM ALLITEMS';
EXEC (@v_sqlStatement); --returns a resultset/cursor, just like calling SELECT 

В Oracle мне нужно использовать блок PL / SQL:

SET AUTOPRINT ON;
DECLARE
 V_Sqlstatement Varchar2(2000);
 outputData SYS_REFCURSOR;
BEGIN
 V_Sqlstatement := 'SELECT * FROM ALLITEMS';
 OPEN outputData for v_Sqlstatement; 
End;
--result is : anonymous block completed

Но все, что я получаю, это «анонимный блок завершен».
Как мне заставить его вернуть курсор?
(Я знаю, что если я сделаю AUTOPRINT, он напечатает информацию в REFCURSOR (это не печатает в коде выше, но это другая проблема))

Я буду вызывать этот динамический SQL из кода (ODBC, C ++), и мне нужно, чтобы он возвращал курсор.
Как мне это сделать? Я в тупике.

Ответы [ 4 ]

39 голосов
/ 28 января 2010

Вы можете написать функцию PL / SQL для возврата этого курсора (или вы можете поместить эту функцию в пакет, если у вас есть больше кода, связанного с этим):

CREATE OR REPLACE FUNCTION get_allitems
  RETURN SYS_REFCURSOR
AS
  my_cursor SYS_REFCURSOR;
BEGIN
  OPEN my_cursor FOR SELECT * FROM allitems;
  RETURN my_cursor;
END get_allitems;

Это вернет курсор.

Убедитесь, что ваша строка SELECT не помещается в кавычки в PL / SQL, когда это возможно. Если поместить его в строки, это означает, что он не может быть проверен во время компиляции и что его нужно анализировать всякий раз, когда вы его используете.


Если вам действительно нужно использовать динамический SQL, вы можете поместить свой запрос в одинарные кавычки:

  OPEN my_cursor FOR 'SELECT * FROM allitems';

Эта строка должна анализироваться при каждом вызове функции, которая обычно будет медленнее и скрывает ошибки в вашем запросе до времени выполнения.

Обязательно используйте переменные связывания, где это возможно, чтобы избежать жестких разборов :

  OPEN my_cursor FOR 'SELECT * FROM allitems WHERE id = :id' USING my_id;
8 голосов
/ 28 января 2010

в SQL * Plus вы также можете использовать переменную REFCURSOR:

SQL> VARIABLE x REFCURSOR
SQL> DECLARE
  2   V_Sqlstatement Varchar2(2000);
  3  BEGIN
  4   V_Sqlstatement := 'SELECT * FROM DUAL';
  5   OPEN :x for v_Sqlstatement;
  6  End;
  7  /

ProcÚdure PL/SQL terminÚe avec succÞs.

SQL> print x;

D
-
X
1 голос
/ 28 января 2010

Вы должны быть в состоянии объявить курсор как переменную связывания (называемые параметрами в других СУБД)

как писал Винсент, вы можете сделать что-то вроде этого:

begin
  open :yourCursor
    for 'SELECT "'|| :someField ||'" from yourTable where x = :y'
      using :someFilterValue;
end;

Вы должны привязать 3 переменные к этому сценарию. Строка ввода для "someField", значение для "someFilterValue" и курсор для "yourCursor", который должен быть объявлен как выходная переменная.

К сожалению, я понятия не имею, как бы вы сделали это с C ++. (Хотя, к счастью для меня, можно сказать. ;-))

В зависимости от того, какую библиотеку доступа вы используете, это может быть королевская боль или прямолинейность.

0 голосов
/ 02 августа 2014

Этот параметр должен быть установлен:

SET SERVEROUTPUT ON 
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...