梧桐数据库的行转列

简介

梧桐数据库的两种行转列写法:

  1. group by + sum + case when
  2. group by + string_agg + split_part(分组,行转列,字符切割)

环境准备

--建表
CREATE TABLE public.sales (
	"year" int4 NOT NULL,
	quarter int4 NOT NULL,
	sales_amount numeric NULL
);

--插入数据
INSERT INTO public.sales ("year", quarter, sales_amount) VALUES(2018, 1, 100);
INSERT INTO public.sales ("year", quarter, sales_amount) VALUES(2018, 2, 200);
INSERT INTO public.sales ("year", quarter, sales_amount) VALUES(2018, 3, 300);
INSERT INTO public.sales ("year", quarter, sales_amount) VALUES(2018, 4, 400);
INSERT INTO public.sales ("year", quarter, sales_amount) VALUES(2019, 1, 500);
INSERT INTO public.sales ("year", quarter, sales_amount) VALUES(2019, 2, 600);
INSERT INTO public.sales ("year", quarter, sales_amount) VALUES(2019, 3, 700);
INSERT INTO public.sales ("year", quarter, sales_amount) VALUES(2019, 4, 800);

我们想将每个季度的销售额作为一列,年份作为行,结果:

 year | q1  | q2  | q3  | q4
------+-----+-----+-----+-----
 2018 | 100 | 200 | 300 | 400
 2019 | 500 | 600 | 700 | 800
(2 rows)

方法1:使用group by + sum + case when

使用GROUP BY + SUM + CASE WHEN 也可以实现将行转列的效果。以下是一个示例:

SELECT  
    year,  
    SUM(CASE WHEN quarter = 1 THEN sales_amount ELSE 0 END) AS q1,  
    SUM(CASE WHEN quarter = 2 THEN sales_amount ELSE 0 END) AS q2,  
    SUM(CASE WHEN quarter = 3 THEN sales_amount ELSE 0 END) AS q3,  
    SUM(CASE WHEN quarter = 4 THEN sales_amount ELSE 0 END) AS q4  
FROM  
    sales  
GROUP BY  
    year  
ORDER BY  
    year;

在这个示例中,我们使用了四个不同的CASE WHEN表达式来计算每个季度的销售额。在每个CASE WHEN表达式中,我们检查季度是否等于1、2、3或4,如果是,就将对应的销售额加入到该季度的总计中。否则,我们将0加入到总计中。

在查询中,我们使用GROUP BY子句对年份进行分组,并对每个季度的销售额进行求和。结果与使用crosstab函数得到的结果相同。

方法2:使用group by + string_agg + split_part(分组,行转列,字符切割)

使用GROUP BY + string_agg + split_part 也可以实现将行转列的效果。以下是一个示例:

SELECT  
    year,  
    split_part(sales_agg, ',', 1)::numeric AS q1,  
    split_part(sales_agg, ',', 2)::numeric AS q2,  
    split_part(sales_agg, ',', 3)::numeric AS q3,  
    split_part(sales_agg, ',', 4)::numeric AS q4  
FROM (  
    SELECT  
        year,  
        string_agg(sales_amount::text, ',' ORDER BY quarter) AS sales_agg  
    FROM sales  
    GROUP BY year  
) AS sales_pivot;

在这个示例中,我们使用string_agg函数将每个季度的销售额连接成一个以逗号分隔的字符串(这里一定需要加上order by子句)。然后,我们使用split_part函数将字符串拆分成四个部分,以获取每个季度的销售额,并将其转换为数字类型。最后,我们在外部查询中指定了每个季度的数据类型和名称。

在查询中,我们首先使用GROUP BY子句对年份进行分组,并使用string_agg函数将每个季度的销售额连接成一个以逗号分隔的字符串。然后,我们在外部查询中使用split_part函数将字符串拆分成四个部分,并将其转换为数字类型,以获取每个季度的销售额。结果与使用crosstab函数或GROUP BY + SUM + CASE WHEN得到的结果相同。

其他数据库比较

Oracle和Mysql可以使用通用的条件聚合而不是尝试拆分字符串

SELECT  
    year,  
    SUM(CASE WHEN quarter = 1 THEN sales_amount ELSE 0 END) AS q1,  
    SUM(CASE WHEN quarter = 2 THEN sales_amount ELSE 0 END) AS q2,  
    SUM(CASE WHEN quarter = 3 THEN sales_amount ELSE 0 END) AS q3,  
    SUM(CASE WHEN quarter = 4 THEN sales_amount ELSE 0 END) AS q4  
FROM  
    sales  
GROUP BY  
    year  
ORDER BY  
    year;
相关推荐
幽兰的天空31 分钟前
python实现excel数据导入数据库
数据库
尘佑不尘2 小时前
shodan5,参数使用,批量查找Mongodb未授权登录,jenkins批量挖掘
数据库·笔记·mongodb·web安全·jenkins·1024程序员节
传输能手2 小时前
从三方云服务器将数据迁移至本地,如何保障安全高效?
大数据·服务器·数据库
BinTools图尔兹2 小时前
CQ社区版 v2024.10 | 支持k8s、helm部署!
数据库·安全·k8s·helm·数据安全·数据库管理员
北笙··3 小时前
Redis慢查询分析优化
数据库·redis·缓存
p-knowledge3 小时前
redis的三种客户端
数据库·redis·缓存
积水成江3 小时前
Redis相关面试题
数据库·redis·缓存
bigcarp3 小时前
Django ORM 数据库管理 提高查询、更新性能的技巧和编程习惯:
数据库·python·django
Zilliz Planet4 小时前
GenAI 生态系统现状:不止大语言模型和向量数据库
数据库·人工智能·语言模型·自然语言处理
瓜牛_gn5 小时前
redis详细教程(4.GEO,bitfield,Stream)
数据库·redis·缓存