在视图列中包含一组行

时间:2011-08-17 17:45:12

标签: sql postgresql view common-table-expression

设计: 一个主表,其中的每个条目可以具有“已检查”的一组选项中的零个或多个。在我看来,如果选项是单独的表的一部分并且在主表和选项表之间进行映射,那么维护(添加/删除选项)会更容易。

目标: 包含主表信息的视图,以及该行已映射到的所有选项。但是后一种信息存在于视图中,应该可以轻松地提取选项的ID及其描述。

下面的实现特定于PostgreSQL,但任何适用于数据库的范例都很有用。

执行我想要的选择语句是:

WITH tmp AS (
        SELECT
            tmap.MainID AS MainID,
            array_agg(temp_options) AS options
        FROM tstng.tmap
        INNER JOIN (SELECT id, description FROM tstng.toptions ORDER BY description ASC) AS temp_options
            ON tmap.OptionID = temp_options.id
        GROUP BY tmap.MainID
    )
SELECT tmain.id, tmain.contentcolumns, tmp.options
FROM tstng.tmain
INNER JOIN tmp
    ON tmain.id = tmp.MainID;

但是,尝试从此select语句创建视图会生成错误: 列“选项”具有伪类型记录[]

我发现的解决方案是将选项数组(record [])转换为文本数组(text [] []);但是,我很想知道那里是否有更好的解决方案。 作为参考,创建指令:

CREATE OR REPLACE VIEW tstng.vsolution AS
    WITH tmp AS (
            SELECT
                tmap.MainID AS MainID,
                array_agg(temp_options) AS options
            FROM tstng.tmap
            INNER JOIN (SELECT id, description FROM tstng.toptions ORDER BY description ASC) AS temp_options
                ON tmap.OptionID = temp_options.id
            GROUP BY tmap.MainID
        )
    SELECT tmain.id, tmain.contentcolumns, CAST(tmp.options AS text[][])
    FROM tstng.tmain
    INNER JOIN tmp
        ON tmain.id = tmp.MainID;

最后,DDL以防我的描述不清楚:

CREATE TABLE tstng.tmap (
        mainid INTEGER NOT NULL,
        optionid INTEGER NOT NULL
    );

CREATE TABLE tstng.toptions (
        id INTEGER NOT NULL,
        description text NOT NULL,
        unwanted_column text
    );

CREATE TABLE tstng.tmain (
        id INTEGER NOT NULL,
        contentcolumns text
    );

ALTER TABLE tstng.tmain ADD CONSTRAINT main_pkey PRIMARY KEY (id);
ALTER TABLE tstng.toptions ADD CONSTRAINT toptions_pkey PRIMARY KEY (id);
ALTER TABLE tstng.tmap ADD CONSTRAINT tmap_pkey PRIMARY KEY (mainid, optionid);
ALTER TABLE tstng.tmap ADD CONSTRAINT tmap_optionid_fkey FOREIGN KEY (optionid)
    REFERENCES tstng.toptions (id);
ALTER TABLE tstng.tmap ADD CONSTRAINT tmap_mainid_fkey FOREIGN KEY (mainid)
    REFERENCES tstng.tmain (id);

1 个答案:

答案 0 :(得分:1)

您可以创建复合类型,例如temp_options_type with:

DROP TYPE IF EXISTS temp_options_type;
CREATE TYPE temp_options_type AS (id integer, description text);

之后,只需在temp_options内将array_agg投射到该类型,因此它会返回temp_options_type[]而不是record[]

DROP VIEW IF EXISTS tstng.vsolution;
CREATE OR REPLACE VIEW tstng.vsolution AS
    WITH tmp AS
    (
        SELECT
            tmap.MainID AS MainID,
            array_agg(CAST(temp_options AS temp_options_type)) AS options
        FROM
            tstng.tmap INNER JOIN
            (
                SELECT id, description
                FROM tstng.toptions
                ORDER BY description
            ) temp_options
            ON tmap.OptionID = temp_options.id
        GROUP BY tmap.MainID
    )
    SELECT tmain.id, tmain.contentcolumns, tmp.options
    FROM tstng.tmain
    INNER JOIN tmp ON tmain.id = tmp.MainID;

示例结果:

TABLE tstng.vsolution;
 id | contentcolumns |        options        
----+----------------+-----------------------
  1 | aaa            | {"(1,xxx)","(2,yyy)"}
  2 | bbb            | {"(3,zzz)"}
  3 | ccc            | {"(1,xxx)"}
(3 rows)