ORACLE行转列、列转行实现方式及案例
行转列
案例
假设我们有一个名为sales的表,其中包含了产品销售数据。表中有三列:product(产品名称)、year(年份)和amount(销售额)。表中的数据如下:
将这个表中的数据进行行转列,使得每一行表示一个产品,每一列表示一年的销售额。
使得得到以下结果:
方式1.PIVOT
PIVOT是Oracle 11g之后引入的一种用于行列转换的函数。它可以将查询结果中的行数据转换为列数据,从而实现行列转换。PIVOT函数的基本语法如下:
SELECT ... FROM ... PIVOT (aggregate_function(column_to_aggregate) FOR column_to_pivot IN (list_of_values))
其中,aggregate_function是一个聚合函数,如SUM、MAX、MIN等;column_to_aggregate是要进行聚合的列;column_to_pivot是要进行行列转换的列;list_of_values是要转换为列的值的列表。
实现案例所示效果可以通过如下方式:
SELECT * FROM sales PIVOT (SUM(amount) FOR year IN (2018, 2019, 2020));
方式2.MAX和DECODE
DECODE 可以根据条件返回不同的值。DECODE 函数的基本语法如下:
DECODE(expression, search1, result1, search2, result2, ..., default)
其中,expression 是要进行比较的表达式;search1、search2 等是要进行比较的值;result1、result2 等是当表达式与对应的搜索值相等时返回的结果;default 是当表达式与所有搜索值都不相等时返回的默认值。
若要实现案例的效果可以通过以下方式:
SELECT product, MAX(DECODE(year, 2018, amount)) AS "2018", MAX(DECODE(year, 2019, amount)) AS "2019", MAX(DECODE(year, 2020, amount)) AS "2020" FROM sales GROUP BY product;
注意:这里取的是每年数据的最大值,每年每产品指只会一条,而方式3是取总和
方式3.CASE WHEN和GROUP BY
SELECT product, SUM(CASE WHEN year = 2018 THEN amount END) AS "2018", SUM(CASE WHEN year = 2019 THEN amount END) AS "2019", SUM(CASE WHEN year = 2020 THEN amount END) AS "2020" FROM sales GROUP BY product;
注意:这里取的是每年数据的总和,每年每产品的总和,而方式2是取最大的一条
列转行
案例
与行转列的案例相反假设我们有一个名为sales的表,其中包含了产品销售数据。表中有四列:product(产品名称)、2018(2018年销售额)、2019(2019年销售额)和2020(2020年销售额)。表中的数据如下:
现在我们想要将这个表中的数据进行列转行,使得每一行表示一个产品在某一年的销售额。
实现效果如下:
方式1.UNPIVOT
UNPIVOT是Oracle中用于实现列转行的函数,基本语法如下:
SELECT ... FROM ... UNPIVOT (column_to_unpivot FOR new_column_name IN (list_of_columns))
其中,column_to_unpivot是要进行列转行的列;new_column_name是新生成的列的名称;list_of_columns是要转换为行的列的列表。
实现案例中的结果可以用如下方式:
SELECT * FROM sales UNPIVOT (amount FOR year IN ("2018", "2019", "2020"));
方式2.UNION ALL
SELECT product, '2018' AS year, "2018" AS amount FROM sales UNION ALL SELECT product, '2019' AS year, "2019" AS amount FROM sales UNION ALL SELECT product, '2020' AS year, "2020" AS amount FROM sales;
注意:单引号表示的是固定字符2018即字段名2018,双引号表示字段2018下的值