列转行
2025年4月23日大约 1 分钟
列转行
select
WW_XH,
WW_LB,
REGEXP_COUNT(WW_LB, '[^,]+') ,
level as lb
FROM T_XXBS
connect by level <= REGEXP_COUNT(WW_LB, '[^,]+') and WW_XH = prior WW_XH and prior dbms_random.value is not null
关于进行行转列时使用的sql语句:
connect by level <= regexp_count(temp.col, '[^、]+')
and temp.id= prior temp.id
and prior dbms_random.value is not null
关于为什么使用上述语句:
CONNECT BY吧?它相当于是一个递归的自连接,不断地把每层的连接结果叠加到结果集中。两层之间的连接条件和递归出口写在CONNECT BY中。在这里我们的数据并无父子关系,只是要让同一行数据重复出现,因此我们的连接的条件只用到了表的主键id=PRIOR id, 此外再用LEVEL控制层数作为递归出口。但ORACLE有个检查,如果你有前后连接条件(id=PRIOR id),但是同一行数据再次出现,它就会报一个错:
ERROR:
ORA-01436: CONNECT BY loop in user data
为了欺骗它,这里用了一个PRIOR DBMS_RANDOM.VALUE, 因为DBMS_RANDOM.VALUE每次调用都返回不同结果,所以它认为两行数据不一样,所以不报错了。
行专列
SELECT *
FROM student
PIVOT (
SUM(score) FOR subject IN (语文, 数学, 英语)
)
SELECT name,
MAX(
CASE
WHEN subject='语文'
THEN score
ELSE 0
END) AS "语文",
MAX(
CASE
WHEN subject='数学'
THEN score
ELSE 0
END) AS "数学",
MAX(
CASE
WHEN subject='英语'
THEN score
ELSE 0
END) AS "英语"
FROM student
GROUP BY name