5

我正在使用以下查询在Oracle 11g (11.2.0.3.0)中创建视图。

CREATE OR REPLACE FORCE VIEW V_DOCUMENTS_LIST
(
   ID_DOC,
   ATTACHMENTS_COUNT,
   TOTAL_DIMENSION,
   INSERT_DATE,
   ID_STATE,
   STATE,
   ID_INSTITUTE,
   INSTITUTE,
   HASJOB
)
AS
    SELECT D.ID_DOC,
        COUNT (F.ID_FILE) AS ATTACHMENTS_COUNT,
        CASE
           WHEN SUM (F.DIMENSION) IS NULL THEN 0
           ELSE SUM (F.DIMENSION)
        END
           AS TOTAL_DIMENSION,
        D.INSERT_DATE,
        D.ID_STATE,
        S.STATE_DESC AS STATE,
        D.ID_INSTITUTE,
        E.NAME AS INSTITUTE,
        CASE
           WHEN EXISTS (SELECT D.ID_DOC FROM JOB) THEN 'true'
           ELSE 'false'
        END
           AS HASJOB
    FROM DOCUMENTS D
        LEFT JOIN FILES F ON D.ID_DOC = F.ID_DOC
        JOIN STATES S ON D.ID_STATE = S.ID_STATE
        JOIN INSTITUTES E ON D.ID_INSTITUTE = E.ID_INSTITUTE
    GROUP BY D.ID_DOC,
        D.INSERT_DATE,
        D.ID_STATE,
        S.STATE_DESC,
        D.ID_INSTITUTE,
        E.NAME;

然后我查询该视图以获取页面中 a 的DataGridViewASPX

SELECT * 
FROM V_DOCUMENTS_LIST
ORDER BY ID_STATE DESC, INSTITUTE, INSERT_DATE DESC;

相关表和关系

文件;文件;工作;

文件 (1-1) <----> (0-N) 文件

工作 (0-1) <----> (0-N) 文件

查询视图,我得到完整的文档列表及其所有相关信息(ID、描述、日期、状态等)以及每个文档:

  • 附件总数
  • 附加文件的总尺寸(以字节为单位)
  • 布尔值,指示是否至少有一个JOB 与 相关联DOCUMENT

在视图包含几千条记录之前,一切正常。现在记录数量正在增加,SELECT * FROM视图上大约需要 2:30 分钟,有 15.000-20.000 条记录。我知道我的观点中一个非常耗时的部分是嵌套的SELECT

CASE
    WHEN EXISTS (SELECT D.ID_DOC FROM JOB) THEN 'true'
    ELSE 'false'
END
AS HASJOB

如何优化我的视图?

4

2 回答 2

7

要解决不存在的问题,您可以添加一个联接:

LEFT JOIN (select distinct id_doc from JOB) J
ON d.id_doc = J.id_doc

Has_job 列将是:

    CASE
       WHEN j.id_doc is not null THEN 'true'
       ELSE 'false'
    END AS HASJOB

PS:您当前的实现存在问题,SELECT D.ID_DOC FROM JOB如果作业表有行,则始终包含行。它与 等价select * from job,因为存在只是测试行的存在。一个逻辑上正确的实现是:SELECT 1 FROM JOB j where j.id_doc = D.ID_DOC.

于 2015-11-20T14:10:49.463 回答
1

您将在 table 上进行完整索引JOB,将WHERE子句放入查询中:

SELECT D.ID_DOC FROM JOB
于 2015-11-20T14:12:18.010 回答