数据库的行列转换问题

在平常的工作中或者面试中,我们可能有遇到过数据库的行列转换问题。今天我们就来讨论下。

1.创建表

首先我们来创建一张表。

sql语句:

--1. 创建数据表
if OBJECT_ID(‘Score‘) is not null drop table Score

create table Score
(
    姓名 nvarchar(128),
    课程 nvarchar(128),
    分数 int
)

insert into Score values(‘张三‘,‘语文‘,98)
insert into Score values(‘张三‘,‘数学‘,89)
insert into Score values(‘张三‘,‘物理‘,78)
insert into Score values(‘李四‘,‘语文‘,79)
insert into Score values(‘李四‘,‘数学‘,88)
insert into Score values(‘李四‘,‘物理‘,100)

select * from Score

执行结果:

2. 传统的行列转换

2.1 纵表转横表

先看看我们要转成的横表张什么样子:

既然这个表只有两列,那么可以根据姓名进行分组。先把姓名拼凑出来,后面的分数我们再想办法。

sql:

select t.姓名 2 from Score as t 3 group by t.姓名

结果:

分析:

  1. 我们先拿到语文这个科目的分数。既然我们用到了group by 语句,这里肯定要用聚合函数来求分数。
  2. 而且我们只需要语文这一科的成绩,分组出来的 一共有 3列 ,分别是 语文、数学、物理  。  那么就需要判断科目来取分数。

这里符合我们需求的 case 语句就登场了。他和c#中switch-case 作用一样。

sql case 语句语法:

case 字段
    when 值1 then 结果
    when 值2 then 结果2
    ...
    else 默认结果
end

求语文的分数就简单了:

select t.姓名,
SUM(case t.课程 when ‘语文‘ then t.分数 else 0 end) as 语文
from Score as t
group by t.姓名

结果:

既然语文的分数取到了,其他科目改变下条件就可以了。

完整的sql:

select t.姓名,
SUM(case t.课程 when ‘语文‘ then t.分数 else 0 end) as 语文,
SUM(case t.课程 when ‘数学‘ then t.分数 else 0 end) as 数学,
SUM(case t.课程 when ‘物理‘ then t.分数 else 0 end) as 物理
from Score as t
group by t.姓名

OK,到这儿,我们传统方式的纵表转横表就大功告成了。

2.2 横表转纵表

那么我们可以把转换过来的横表再转换回去吗?

我们先把刚刚转好的表,插入一个新表ScoreHb 中。

-- 转换的表插入新表
select t.姓名,
SUM(case t.课程 when ‘语文‘ then t.分数 else 0 end) as 语文,
SUM(case t.课程 when ‘数学‘ then t.分数 else 0 end) as 数学,
SUM(case t.课程 when ‘物理‘ then t.分数 else 0 end) as 物理
into ScoreHb
from Score as t
group by t.姓名

这时ScoreHb 就是我们刚转换好的横表,我们再想办法把他转回来。

怎么转呢? 一步步来。我们也先把张三和李四的语文成绩查出来。

sql:

 --张三李四语文的分数
 select t.姓名,
 ‘语文‘ as 课程,
 t.语文 as 分数
 from ScoreHb as t

结果:

还有两科的数据怎么办呢? 很简单,我们一个个都查出来,然后用 union all 把他们组合为一张表就可以了。

sql:

-- union all链接3个科目
select t.姓名,
‘语文‘ as 课程,
t.语文 as 分数
from ScoreHb as t
union all
select t.姓名,
‘数学‘ as 课程,
t.数学 as 分数
from ScoreHb as t
union all
select t.姓名,
‘物理‘ as 课程,
t.物理 as 分数
from ScoreHb as t
order by t.姓名 desc

结果:

这样,我们就把表又变回去了。

但是大家有没有觉得很麻烦呢?别急,我们有更简单的办法。下面为大家介绍pivot关系运算符。

3. 用pivot和unPIVOT运算符进行转换

pivot是sql server 2005 提供的运算符,所以只要数据库在05版本以上的都可以使用。主要用于行和列的转换。

3.1 pivot纵表转横表

sql:

select
    t2.姓名,
    t2.数学,
    t2.物理,
    t2.语文
from Score as t1
pivot (sum(分数) for 课程 in(数学,语文,物理)) as t2

结果:

是不是代码简洁多了。

pivot将原来表中 课程字段中的 数据行 数学,语文,物理 转换为列,并用sum取对应列的值。

我们只需要记住它的用法就可以了。

3.2 unpivot 横表转纵表

既然有privot可以纵表转横表。那么有没有运算符帮我们转回来呢?

答案是肯定的,他就是unpivot

sql:

 select
     *
 from
 ScoreHb
 unpivot (分数 for 课程 in (语文,数学,物理)) as t4

结果:

 unpivot 将 语文,数学,物理 列转为行,分数为新的一列存放对应的值。

是不是比我们之前一个个表查询拼接,方便了很多。

时间: 2024-11-02 15:04:19

数据库的行列转换问题的相关文章

通过sql做数据透视表,数据库表行列转换(pivot和Unpivot用法)(一)

在mssql中大家都知道可以使用pivot来统计数据,实现像excel的透视表功能 一.MSsqlserver中我们通常的用法 1.Sqlserver数据库测试 ---创建测试表 Create table s( [name] nvarchar(50), book nvarchar(50), saledNumber int ) ----插入测试数据 insert into s ([name],book,saledNumber) values('小王','java从入门到精通',10); inser

简单的数据库表行列转换实例

原表如下: 列转行: select 姓名, MAX(CASE 课程 when '语文' then 分数 else 0 END) 语文, MAX(CASE 课程 when '数学' then 分数 else 0 END) 数学, MAX(CASE 课程 when '物理' then 分数 else 0 END) 物理from 成绩 GROUP BY 姓名 得到如下结果: 反过来,行转列: select * from ( select 姓名,'语文' 课程, 语文 分数 from 成绩2 UNION

数据库类似行列转换SQL

1.原表test结构如下: 2.我们想要得到的结果为: 3.第一种方法:静态方法,即列固定 SQL如下: select * from  ( SELECT date,name,sum(num) num FROM test  GROUP BY date,name ) t pivot ( sum(num) for t.name in ([alan],[andy],[baicl])) as a 4.第二种方法:动态方法,即列为通过SQL查询获取 SQL如下: declare @sql nvarchar(

一道面试题引发的数据库行列转换实践

问题场景 最近有个朋友去面试,问了我一道面试题.题目如下,在形如下面的数据库表score中,找出每门成绩(grade)都大于等于80分的学生姓名. ---------------------------------------- name     | course  | grade ---------------------------------------- zhangsan  | Java  | 70 ---------------------------------------- zha

SQL行列转换6种方法

在进行报表开发时,很多时候会遇到行列转换操作,很对开发人员针对于SQL级别行列转换操作一直不甚理解,今天正好抽空对其进行了一些简单的总结.这里主要列举3种可以实现SQL行列转换的方法,包括通用SQL解法以及Oracle支持解法. 一.测试数据 测试表依旧采用Oracle经典的scott模式下的dept表和emp表,结构如下: DEPT: create table DEPT ( DEPTNO NUMBER(2) not null, DNAME VARCHAR2(14), LOC VARCHAR2(

Mysql实现行列转换

前言: 最近又玩起了sql语句,想着想着便给自己出了一道题目:“行列转换”.起初瞎折腾了不少时间也上网参考了一些博文,不过大多数是采用oracle数据库当中的一些便捷函数进行处理,比如”pivot”.那么,在Mysql环境下如何处理? 自己举了个小例子: sql代码实现: 1 -- Step1:建表并插入数据 2 3 -- Step2:中间转换,即“二维转一维”得到一维表. 4 5 -- Step3:利用IF判断并“group by”即可得到目标结果表. 6 7 8 9 -- 加载 10 11

锋利的SQL-SQL Server的表旋转(行列转换)

所谓表旋转,就是将表的行转换为列,或是将表的列转换为行,这是从SQL Server 2005开始提供的新技术.因此,如果希望使用此功能,需要将数据库的兼容级别设置为90.表旋转在某些方面也是解决了表的数据存储和实际需要之间的矛盾.例如,图9-4所示的是一个典型的产品销售统计表,这种格式虽然便于阅读,但是在进行数据表存储的时候却并不容易管理,产品销售数据表通常需要设计成图9-5所示的结构.这样就带来一个问题,用户既希望数据容易管理,又希望能够生成一种能够容易阅读的表格数据,这时候就可以使用表旋转技

SQL Server中行列转换 Pivot UnPivot

SQL Server中行列转换 Pivot UnPivot PIVOT用于将列值旋转为列名(即行转列),在SQL Server 2000可以用聚合函数配合CASE语句实现 PIVOT的一般语法是:PIVOT(聚合函数(列) FOR 列 in (-) )AS P 完整语法: table_source PIVOT( 聚合函数(value_column) FOR pivot_column IN(<column_list>) ) UNPIVOT用于将列明转为列值(即列转行),在SQL Server 2

ABAP 内表的行列转换-发货通知单-打印到Excel里

需要传入数据到Excel里的模板如上图所示 *&---------------------------------------------------------------------* *& Report  Z_TEST_COL_TO_ROW_02 *& *&---------------------------------------------------------------------* *& *& *&----------------