2

一些背景:数据库是甲骨文。我正在尝试为表格的每一行创建一个分隔字符串。分隔字符串需要包含项目的两个类别(如果可用,则始终至少有一个类别)。有 3 个表,ITEM、ITEM_CAT 和 ITEM_ITEM_CAT。ITEM 表包含所有项目。ITEM_CAT 表包含所有可能的项目类别。ITEM_ITEM_CAT 包含项目 ID 和类别 ID 之间的所有映射,本质上是一个键值表。

我创建了下面的 SQL,它能够获取特定项目的分隔字符串,但我需要一个可以针对整个表运行的查询。

SELECT 'ITEM'||'%#'|| outerTable.ITEM_ID ||'%#'|| 
    (SELECT midTable.item_cat_nam
     FROM
            (SELECT innerTable.item_cat_nam AS item_cat_nam, innerTable.item_id AS item_id, ROWNUM AS rn
             FROM
                    (SELECT ic.ITEM_CAT_NAM AS item_cat_nam, i.ITEM_ID AS item_id
                     FROM ITEM_CAT ic, ITEM_ITEM_CAT iic, ITEM i
                     WHERE i.ITEM_ID = iic.ITEM_ID 
                     AND iic.ITEM_CAT_CD = ic.ITEM_CAT_CD
                     AND 287484 = i.item_id
                    ) innerTable
            ) midTable
    WHERE rn = 1
    ) ||'%#'||
    (SELECT midTable.item_cat_nam
     FROM
            (SELECT innerTable.item_cat_nam AS item_cat_nam, innerTable.item_id AS item_id, ROWNUM AS rn
             FROM
                    (SELECT ic.ITEM_CAT_NAM AS item_cat_nam, i.ITEM_ID AS item_id
                     FROM ITEM_CAT ic, ITEM_ITEM_CAT iic, ITEM i
                     WHERE i.ITEM_ID = iic.ITEM_ID 
                     AND iic.ITEM_CAT_CD = ic.ITEM_CAT_CD
                     AND 287484 = i.item_id
                    ) innerTable
            ) midTable
    WHERE rn = 2
    )
FROM OFR outerTable
WHERE outerTable.ITEM_ID = 287484;

我需要能够将外部表的 ITEM_ID 向下传递到最后一个内部连接。当我只需要类别时(通过下面的 SQL 语句,只需要一个内连接),但引入多个类别时,我可以这样做;我需要 rownum (以获得多个类别),然后需要更多的内部连接,我似乎无法将 ITEM_ID 向下传递超过一个内部连接,这就是问题所在......

SELECT 'ITEM'||'%#'|| outerTable.OFR_ID ||'%#'|| 
    (SELECT ic.ITEM_CAT_NAM
     FROM ITEM_CAT ic, ITEM_ITEM_CAT iic, ITEM i
     WHERE i.ITEM_ID = iic.ITEM_ID 
     AND iic.ITEM_CAT_CD = ic.ITEM_CAT_CD
     AND outerTable.OFR_ID = i.item_id
             AND rownum = 1
    ) innerTable
FROM OFR outerTable;

有人能帮忙吗?

预先感谢您的任何帮助。

4

1 回答 1

2

不用担心。你需要这样的东西...

SELECT 'ITEM' || '%#' || Item_ID || '%#' || CatName1 || '%#' || CatName2
FROM outerTable
INNER JOIN (
  SELECT
    Item_ID,
    MAX(CASE WHEN rn = 1 THEN Item_Cat_Nam ELSE NULL END) CatName1,
    MAX(CASE WHEN rn = 2 THEN Item_Cat_Nam ELSE NULL END) CatName2
  FROM (
    SELECT
      Item_ID,
      Item_Cat.Item_Cat_Nam,
      ROW_NUMBER() OVER (PARTITION BY Item_ID ORDER BY Item_ID) rn
    FROM Item
    INNER JOIN Item_Item_Cat USING (Item_ID)
    INNER JOIN Item_Cat USING (Item_Cat_Cd)
  ) GROUP BY Item_ID
) USING (Item_ID)

最里面的查询使用 ROW_NUMBER 函数将 1、2、3 等分配给为每个项目找到的每个类别。PARTITION BY 将每个项目的编号从 1 重新开始。ORDER BY 是必需的,所以我使用了 Item_ID,因为嘿,为什么不呢?如果您有一个首选列,只需使用它即可 - 它将用于分配行号。内部查询将输出如下内容:

Item_ID  Item_Cat_Nam  rn
-------  ------------  --
      1  Category aa    1
      1  Category xy    2
      1  Category ef    3
      2  Category xy    1
      2  Category ax    2
      3  Category ef    1

围绕最内层查询的查询使用 MAX 将rn每个 Item_ID 的前两个值展平为一行。rn=1 的 Item_Cat_Nam 进入 CatName1 列,而 rn=2 的 Item_Cat_Nam 进入 CatName2 列。当它输入上面显示的结果时,您将得到以下结果:

Item_ID  CatName1     CatName2
-------  -----------  -----------
      1  Category aa  Category xy  (note Category ef is rn=3 so it's ignored)
      2  Category xy  Category ax
      3  Category ef               (note only one row for Item_ID 3)

然后非常外部的查询只是连接所有内容。

另一件事:我使用了“JOIN ... USING”语法,因为在这种情况下,它可以让您消除所有别名(innerTable、i、ic、iic、midTable 等)。这纯粹是因为我对它更舒服,所以它帮助我更快地解决了这个问题。您应该随意使用自己的连接样式 - 毕竟您将成为维护它的人:)

于 2013-03-01T21:39:03.540 回答