Альтернативы для временных таблиц в Oracle - PullRequest
3 голосов
/ 29 сентября 2011
  1. Создайте временную таблицу внутри хранимой процедуры, скажем '#Temp'.
  2. Вставить значения в таблицу «Temp», используя оператор выбора, например. Вставить в #Temp Выберите * из списка сотрудников.
  3. Теперь извлеките данные из этой таблицы Temp, например. Выберите * из #Temp, где # Temp.Id = @id и т. Д.

Как это сделать в Oracle внутри хранимой процедуры ?

Ответы [ 2 ]

15 голосов
/ 29 сентября 2011

Какую бизнес-проблему вы пытаетесь решить? Исключительно редко вам нужно использовать временные таблицы в Oracle. Почему бы тебе просто не

SELECT *
  FROM employees
 WHERE id = p_id_passed_in;

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

В других базах данных вы создаете временные таблицы, потому что вы не хотите делать грязные чтения. Oracle, однако, не допускает грязного чтения. Многовариантная согласованность чтения означает, что Oracle всегда будет показывать вам данные такими, какими они были на момент запуска запроса (или когда транзакция началась, если вы установили уровень изоляции транзакции для сериализуемого). Так что нет необходимости создавать временную таблицу, чтобы избежать грязного чтения.

Если бы вы действительно хотели использовать временные таблицы в Oracle, вы бы не создали таблицу динамически. Вы должны создать глобальную временную таблицу до того, как создадите хранимую процедуру. Структура таблицы будет видна всем сеансам, но данные будут видны только сеансу, который ее вставил. Вы должны заполнить временную таблицу в процедуре, а затем запросить таблицу. Что-то вроде

CREATE GLOBAL TEMPORARY TABLE temp_emp (
  empno number,
  ename varchar2(10),
  job   varchar2(9),
  mgr   number,
  sal   number(7,2)
)
ON COMMIT PRESERVE ROWS;

CREATE OR REPLACE PROCEDURE populate_temp_emp
AS
BEGIN
  INSERT INTO temp_emp( empno,
                        ename,
                        job,
                        mgr,
                        sal )
    SELECT empno, 
           ename,
           job,
           mgr,
           sal
      FROM emp;
END;
/

SQL> begin
  2    populate_temp_emp;
  3  end;
  4  /

PL/SQL procedure successfully completed.

SQL> select *
  2    from temp_emp;

     EMPNO ENAME      JOB              MGR        SAL
---------- ---------- --------- ---------- ----------
      7623 PAV        Dev
      7369 smith      CLERK           7902        800
      7499 ALLEN      SALESMAN        7698       1600
      7521 WARD       SALESMAN        7698       1250
      7566 JONES      MANAGER         7839       2975
      7654 MARTIN     SALESMAN        7698       1250
      7698 BLAKE      MANAGER         7839       2850
      7782 CLARK      MANAGER         7839       2450
      7788 SCOTT      ANALYST         7566       3000
      7839 KING       PRESIDENT                  5000
      7844 TURNER     SALESMAN        7698       1500
      7876 ADAMS      CLERK           7788       1110
      7900 SM0        CLERK           7698        950
      7902 FORD       ANALYST         7566       3000
      7934 MILLER     CLERK           7782       1300
      1234 BAR

16 rows selected.

Как я уже сказал, в Oracle было бы очень необычно хотеть использовать временную таблицу.

2 голосов
/ 29 сентября 2011

Создайте глобальную временную таблицу.

CREATE GLOBAL TEMPORARY TABLE <your_table>
ON COMMIT PRESERVE ROWS   # If needed.  Depends on your needs.
AS SELECT <your_select_query>;

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

http://www.oracle -base.com/articles/8i/TevenTables.php

http://asktom.oracle.com/pls/asktom/f?p=100:11:0::::P11_QUESTION_ID:15826034070548

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