梧桐数据库的行转列

简介

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

  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;
相关推荐
不光头强1 小时前
Spring框架的事务管理
数据库·spring·oracle
百***92023 小时前
【MySQL】MySQL库的操作
android·数据库·mysql
q***76663 小时前
Spring Boot 从 2.7.x 升级到 3.3注意事项
数据库·hive·spring boot
信仰_2739932433 小时前
Redis红锁
数据库·redis·缓存
人间打气筒(Ada)3 小时前
Centos7 搭建hadoop2.7.2、hbase伪分布式集群
数据库·分布式·hbase
心灵宝贝4 小时前
如何在 Mac 上安装 MySQL 8.0.20.dmg(从下载到使用全流程)
数据库·mysql·macos
奋斗的牛马4 小时前
OFDM理解
网络·数据库·单片机·嵌入式硬件·fpga开发·信息与通信
忧郁的橙子.5 小时前
一、Rabbit MQ 初级
服务器·网络·数据库
杰杰7985 小时前
SQL 实战:用户访问 → 下单 → 支付全流程转化率分析
数据库·sql
爬山算法5 小时前
Redis(120)Redis的常见错误如何处理?
数据库·redis·缓存