首页 > 数据库 >PostgreSQL中的行转列

PostgreSQL中的行转列

时间:2023-12-11 15:44:25浏览次数:34  
标签:PostgreSQL agg sales 转列 part year quarter

PG有三种行转列写法:

1、group by + sum + case when

2、用postgresql的crosstab函数

3、group by + string_agg + split_part(分组,行转列,字符切割)

环境准备

 1CREATE TABLE sales (
 2    year INTEGER,
 3    quarter INTEGER,
 4    sales_amount NUMERIC
 5);
 6
 7INSERT INTO sales VALUES
 8    (2018, 1, 100),
 9    (2018, 2, 200),
10    (2018, 3, 300),
11    (2018, 4, 400),
12    (2019, 1, 500),
13    (2019, 2, 600),
14    (2019, 3, 700),
15    (2019, 4, 800);

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

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

方法1:使用crosstab函数

PostgreSQL中可以使用crosstab函数将行转列,需要使用一个额外的模块“tablefunc”,安装完成后,我们可以使用crosstab函数将行转列。

1CREATE EXTENSION tablefunc;
2
3SELECT *
4FROM crosstab(
5    'SELECT year, quarter, sales_amount
6     FROM sales
7     ORDER BY 1, 2',
8    'SELECT quarter FROM generate_series(1,4) AS quarter'
9) AS sales_pivot(year INTEGER, q1 NUMERIC, q2 NUMERIC, q3 NUMERIC, q4 NUMERIC);

在这个例子中,我们将查询作为第一个参数传递给crosstab函数。查询必须按年份和季度排序。第二个参数是一个子查询,用于生成列名。在本例中,我们使用generate_series函数生成1到4之间的数字作为季度列的名称。

crosstab函数将行转换为列,并返回一个新的表。我们在外部查询中指定每个生成的列的数据类型和名称,以便正确返回结果。

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

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

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

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

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

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

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

 1SELECT
 2    year,
 3    split_part(sales_agg, ',', 1)::numeric AS q1,
 4    split_part(sales_agg, ',', 2)::numeric AS q2,
 5    split_part(sales_agg, ',', 3)::numeric AS q3,
 6    split_part(sales_agg, ',', 4)::numeric AS q4
 7FROM (
 8    SELECT
 9        year,
10        string_agg(sales_amount::text, ',' ORDER BY quarter) AS sales_agg
11    FROM sales
12    GROUP BY year
13) AS sales_pivot;

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

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

标签:PostgreSQL,agg,sales,转列,part,year,quarter
From: https://www.cnblogs.com/dancesir/p/17894575.html

相关文章

  • PostgreSQL重命名报错
    ERROR:currentdatabasecannotberenamedSTATEMENT:ALTERDATABASExxxRENAMETOxxx2;查看客户端软件连接信息是否是当前需要修改的数据库,如果是,需要把数据库修改成其他数据库。如,我要修改postgres名称,那么这里就需要把postgres换成其他数据库 ERROR:database......
  • postgresql查看表、字段注释
    一、查看pg表字段‘名称’、‘类型’、‘非空’、‘注释’SELECTa.attnameas字段名,format_type(a.atttypid,a.atttypmod)as类型,a.attnotnullas非空,col_description(a.attrelid,a.attnum)as注释FROMpg_classasc,pg_attributeasawherea.attrelid=c.oida......
  • pg库实现sql行转列
    ......
  • postgresql从入门到精通 - 第37讲:postgres物理备份和恢复概述
       PostgreSQL从小白到专家,是从入门逐渐能力提升的一个系列教程,内容包括对PG基础的认知、包括安装使用、包括角色权限、包括维护管理、、等内容,希望对热爱PG、学习PG的同学们有帮助,欢迎持续关注CUUGPG技术大讲堂。 第37讲:物理备份和恢复概述 第37讲:12月09日(周六)19......
  • PostgreSQL筛选包含中文的字段
    筛选中文内容匹配一个或多个非中文字符SELECT*FROMtable1WHEREfield1~'[^\u4e00-\u9fa5]+';[\u4e00-\u9fa5]表示匹配中文字符,加上前面的取反符号^即表示匹配不是中文字符,+表示一个或多个。\u4e00-\u9fa5是中文字符的Unicode范围。如果想筛选字段不包含中文可以......
  • linux安装postgresql三种方式【转】
    linux下安装PostgreSQL可采用三种方式,二进制已编绎安装包、安装、源码安装三种方式进行安装方式1:yum安装,建议使用这种参看官方文档如下地址:sudoyuminstall-ysudoyuminstall-ypostgresql15-serversudo/usr/pgsql-15/bin/postgresql-15-setupinitdbsudosystemctlena......
  • StackGres 数据库平台工程,使用 Citus + Patroni 创建生产级高可用分布式 PostgreSQL
    系列StackGres,可私有部署的云原生数据库平台工程StackGres 数据库平台工程功能介绍与快速上手StackGres1.6数据库平台工程集群配置管理(K8SPods/PostgreSQL/PgBouncer)StackGres1.6数据库平台工程,集群高可用(Patroni3管理)什么是ShardedCluster(分片集群)Sha......
  • PostgreSQL: select for update实战
    场景需要获取用户申请的流水号,其值记录在number_of_form表中。但当多个用户同时申请时,会出现单号重复的情况,现在需要保证单据号码的一致性解决方案以我搜寻来看,大体有两种做法。悲观锁:总是假设最坏的情况,也就是每次拿数据的时候,都认为别人会修改,所以每次拿数据,都会对符合条......
  • 行转列
    CREATETABLEScores(StudentNameVARCHAR(50),SubjectVARCHAR(50),ScoreINT);INSERTINTOScoresVALUES('张三','语文',90);INSERTINTOScoresVALUES('张三','数学',80);INSERTINTOScoresVALUES('张三'......
  • Github Actions - Creating PostgreSQL service containers
     #Servicecontainerstorunwith`container-job`services:#Labelusedtoaccesstheservicecontainerpostgres:#DockerHubimageimage:postgres#Providethepasswordforpostgresenv:......