gpt4 book ai didi

xml - 带有子查询的 Oracle 数据透视表

转载 作者:数据小太阳 更新时间:2023-10-29 01:48:29 24 4
gpt4 key购买 nike

我在 Oracle PL SQL Developer 中使用 pivot 如下:

SELECT *
FROM population
PIVOT (AVG(Total) for Data_Type IN ('Group1','Group2','Group3'))

这很好用,但我不想每次添加新列或更改列(即 Group4、5、6 等)时都必须进行编辑,所以我尝试了如下子查询:

SELECT *
FROM population
PIVOT (AVG(Total) for Data_Type IN (SELECT Data_Type FROM population))

这会导致以下错误:ORA-00936:缺少表达式。

经过一些研究,我似乎可以用 XML 生成结果,所以我尝试了以下方法:

SELECT *
FROM population
PIVOT XML(AVG(Total) for Data_Type IN (ANY))

这实际上生成了所需的数据,但是是 XML 格式的。所以我的问题是,如何在 PL SQL Developer 中将 XML 结果转换为标准表格格式?或者,如果我想将生成的 XML 文件导入 Crystal Reports 等工具中,我需要为这些结果创建一个模式文件。这是可以在 SQL 中轻松自动生成的内容吗?

最佳答案

您会考虑使用 PIPELINED 函数来实现您的目标吗?

我已经写了一个这样的函数的例子。该示例基于 Tom Kyte 文章中的表格、示例数据和 PIVOT 查询,您可以在他的网站上找到这些文章:

Tom Kyte's article about PIVOT/UNPIVOT

Tom Kyte's article about PIPELINED functions

示例工作如下。

我们创建两种类型:

  • t_pivot_test_obj - 包含我们要从 XML 中检索的列的类型
  • t_pivot_test_obj_tab - 上述对象的嵌套表类型。

然后我们创建一个 PIPELINED 函数,其中包含带有 PIVOT 的查询,它生成 XML(因此您不必对要转换的值进行硬编码)。此函数从生成的 XML 中提取数据,并在生成数据时将 (PIPE) 行传递给调用查询(动态 - 它们不会一次全部生成,这对性能很重要)。

最后,您编写了一个从该函数中选择记录的查询(最后是此类查询的示例)。

CREATE TABLE pivot_test (
id NUMBER,
customer_id NUMBER,
product_code VARCHAR2(5),
quantity NUMBER
);

INSERT INTO pivot_test VALUES (1, 1, 'A', 10);
INSERT INTO pivot_test VALUES (2, 1, 'B', 20);
INSERT INTO pivot_test VALUES (3, 1, 'C', 30);
INSERT INTO pivot_test VALUES (4, 2, 'A', 40);
INSERT INTO pivot_test VALUES (5, 2, 'C', 50);
INSERT INTO pivot_test VALUES (6, 3, 'A', 60);
INSERT INTO pivot_test VALUES (7, 3, 'B', 70);
INSERT INTO pivot_test VALUES (8, 3, 'C', 80);
INSERT INTO pivot_test VALUES (9, 3, 'D', 90);
INSERT INTO pivot_test VALUES (10, 4, 'A', 100);
COMMIT;

CREATE TYPE t_pivot_test_obj AS OBJECT (
customer_id NUMBER,
product_code VARCHAR2(5),
sum_quantity NUMBER
);
/

CREATE TYPE t_pivot_test_obj_tab IS TABLE OF t_pivot_test_obj;
/

CREATE OR REPLACE FUNCTION extract_from_xml RETURN t_pivot_test_obj_tab PIPELINED
AS
v_xml XMLTYPE;
v_item_xml XMLTYPE;
v_index NUMBER;
v_sum_quantity NUMBER;

CURSOR c_customer_items IS
SELECT customer_id, product_code_xml
FROM (SELECT customer_id, product_code, quantity
FROM pivot_test)
PIVOT XML (SUM(quantity) AS sum_quantity FOR (product_code) IN (SELECT DISTINCT product_code
FROM pivot_test));
BEGIN
-- loop through all records returned by query with PIVOT
FOR v_rec IN c_customer_items
LOOP
v_xml := v_rec.product_code_xml;
v_index := 1;

-- loop through all ITEM elements for each customer
LOOP
v_item_xml := v_xml.EXTRACT('/PivotSet/item[' || v_index || ']');

EXIT WHEN v_item_xml IS NULL;

v_index := v_index + 1;

IF v_item_xml.EXTRACT('/item/column[@name="SUM_QUANTITY"]/text()') IS NOT NULL THEN
v_sum_quantity := v_item_xml.EXTRACT('/item/column[@name="SUM_QUANTITY"]/text()').getNumberVal();
ELSE
v_sum_quantity := 0;
END IF;

-- finally, for each customer and item - PIPE the row to the calling query
PIPE ROW(t_pivot_test_obj(v_rec.customer_id,
v_item_xml.EXTRACT('/item/column[@name="PRODUCT_CODE"]/text()').getStringVal(),
v_sum_quantity));
END LOOP;
END LOOP;
END;
/

SELECT customer_id, product_code, sum_quantity
FROM TABLE(extract_from_xml())
;

输出:

CUSTOMER_ID            PRODUCT_CODE SUM_QUANTITY           
---------------------- ------------ ----------------------
1 A 10
1 B 20
1 C 30
1 D 0
2 A 40
2 B 0
2 C 50
2 D 0
3 A 60
3 B 70
3 C 80
3 D 90
4 A 100
4 B 0
4 C 0
4 D 0

16 rows selected

关于xml - 带有子查询的 Oracle 数据透视表,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19298424/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com