梧桐数据库的行转列

简介

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

  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;
相关推荐
背太阳的牧羊人15 分钟前
用于与多个数据库聊天的智能 SQL 代理问答和 RAG 系统(2) —— 从 PDF 文档生成矢量数据库 (VectorDB),然后存储文本的嵌入向量
数据库·人工智能·sql·langchain·pdf
zhangxueyi1 小时前
MySQL之企业面试题:InnoDB存储引擎组成部分、作用
java·数据库·mysql·面试·innodb
代码代码快快显灵1 小时前
Redis 优化秒杀(异步秒杀)
数据库·redis·缓存
极客先躯1 小时前
Redis 安装与配置指南
数据库·redis·数据验证·安装说明·编译和安装·redis 集群配置·查看集群
YaenLi2 小时前
MySQL 安装部署
linux·数据库·mysql
乄北城以北乀2 小时前
一.MySQL程序简介
数据库·mysql
炭烤毛蛋2 小时前
Ubuntu 磁盘修复
linux·数据库·ubuntu
代码代码快快显灵2 小时前
Redis之秒杀活动
数据库·redis·缓存·秒杀活动
一水鉴天4 小时前
为AI聊天工具添加一个知识系统 之27 支持边缘计算设备的资源存储库及管理器
数据库·人工智能·前端框架
拾忆,想起5 小时前
Spring拦截链揭秘:如何在复杂应用中保持控制力
java·数据库·spring