Oracle 中临时表的替代方案

Posted

技术标签:

【中文标题】Oracle 中临时表的替代方案【英文标题】:Alternatives for temporary tables in Oracle 【发布时间】:2011-09-29 14:12:32 【问题描述】:
    在存储过程中创建一个临时表,例如“#Temp”。 使用 select 语句将值插入“Temp”表,例如。插入#Temp 从员工中选择 *。 现在从这个 Temp 表中提取数据,例如。从 #Temp 中选择 *,其中 #Temp.Id = @id 等等。

如何在 Oracle 中的存储过程中做到这一点?

【问题讨论】:

ways to avoid global temp tables in oracle 的可能重复项 【参考方案1】:

您要解决的业务问题是什么?在 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 中实际上想要使用临时表是非常不寻常的。

【讨论】:

感谢您为回答我的问题所做的努力,但这不是我想要的。简而言之,我需要一个替代逻辑来替代 Oracle 中存储的 procs 内的临时表。 @nylon610 - 你能解释一下为什么不能使用全局临时表吗?从技术上讲,使用动态 SQL 在存储过程中创建对象是可行的,但是这种额外的复杂性是值得的,这是非常罕见的。特别是在多用户系统中,因为在 Oracle 中不可能有一个范围仅限于会话的对象。 我在存储过程中有一个选择查询。我必须打印查询结果中的行数,然后查询结果数据本身。我怎样才能做到这一点?全局临时表不适用于我,因为事务结束后不删除,只删除其数据。使用这样的设计是非常危险的。 @meir - 这是一个家庭作业吗?如果“打印”意味着调用dbms_output,那不是您在生产存储过程中执行的操作,因为无法保证调用者能够对结果执行任何操作。客户端打印结果会更明智,因为客户端负责管理表示层并让客户端确定获取的行数。如果您希望返回的行数超过少数,则显示结果然后显示 SQL*Plus 的计数会更有效。【参考方案2】:

创建一个全局临时表。

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/TemporaryTables.php

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

【讨论】:

以上是关于Oracle 中临时表的替代方案的主要内容,如果未能解决你的问题,请参考以下文章

# 实际开发中临时表的使用

# 实际开发中临时表的使用

SQL Server 表变量和临时表的区别

oracle存储过程中临时表的使用,该怎么处理

Oracle存储过程中临时表的使用技巧

oracle存储过程中临时表的使用,该怎么处理