[请教一个比较难搞的sql查询问题]关于合并子表记录成一个字符串和主表列一起显示的查询
主表
c1 c2 c3
1 X Y
2 X Z
子表e1列外键是主表c1列
e0 e1 e2
8 1 你
9 1 好
10 2 吃
11 2 了
12 2 吗
想显示成
1 X Y 你好
2 X Z 吃了吗
------解决方案--------------------SQL 语句本身好像没有合并记录的功能,只能合并列。
一个解决方法是利用函数,如下:
-- 数据准备
CREATE TABLE mt(
c1 NUMBER PRIMARY KEY,
c2 VARCHAR2(30),
c3 VARCHAR2(30));
CREATE TABLE ct(
e0 NUMBER PRIMARY KEY,
e1 NUMBER REFERENCES mt(c1),
e2 VARCHAR2(30));
INSERT INTO mt VALUES(1, 'X ', 'Y ');
INSERT INTO mt VALUES(2, 'X ', 'Z ');
INSERT INTO ct VALUES(8,1, '你 ');
INSERT INTO ct VALUES(9,1, '好 ');
INSERT INTO ct VALUES(10,2, '吃 ');
INSERT INTO ct VALUES(11,2, '了 ');
INSERT INTO ct VALUES(12,2, '吗 ');
-- 创建函数
CREATE OR REPLACE FUNCTION concat_rows(p_parent_id NUMBER) RETURN VARCHAR2 IS
v_result VARCHAR2(120);
BEGIN
FOR rec IN (SELECT e2 FROM ct WHERE e1 = p_parent_id ORDER BY e0) LOOP
v_result := v_result || rec.e2;
END LOOP;
return v_result;
END;
-- SQL 语句
SELECT c1,c2,c3,concat_rows(c1) FROM mt;
-- 输出
C1 C2 C3 CONCAT_ROWS(C1)
---------- ------------------------------ ------------------------------ --------------------------------------------
1 X Y 你好
2 X Z 吃了吗
------解决方案--------------------select t1.c1, replace(substr(MAX(sys_connect_by_path(t1.e2, ', ')),2), ', ', ' ') e2
from
(
select a.*,b.*,row_number()over(partition by a.c1 order by b.e0) rn
from 主表 a,子表 b
where a.c1=b.e1
) t1
start with t1.c1=1
connect by t1.c1 = prior t1.c1
and t1.rn -1 = prior t1.rn
group by t1.c1;