CONNECT BY
требуется, как вы и предполагали.Хитрость заключается в том, чтобы опустить предложение START WITH
, поэтому каждый отдел рассматривается как «корень».Затем мы можем подсчитать сотрудников для каждого «корня», то есть для каждого отдела и всех его подразделений.
Вот и ваш пример.Я также добавил дополнительный уровень в структуру ваших отделов, как более сложный контрольный пример.
WITH dept ( id, name, parent_id) AS (
SELECT 1, 'dep1', null FROM DUAL UNION ALL
SELECT 2, 'dep2', null FROM DUAL UNION ALL
SELECT 3, 'dep21', 2 FROM DUAL UNION ALL
SELECT 4,'dep22', 2 FROM DUAL UNION ALL
SELECT 5, 'dep211', 3 FROM DUAL
),
emp (id, name, dep_id) AS (
SELECT 1, 'Name1', 3 FROM DUAL UNION ALL
SELECT 2, 'Name2', 4 FROM DUAL UNION ALL
SELECT 3, 'Name3', 1 FROM DUAL UNION ALL
SELECT 4, 'Name4', 2 FROM DUAL UNION ALL
SELECT 5, 'Name5', 5 FROM DUAL
),
intermediate as (
select connect_by_root d.name deptname, level lvl, e.id empid, e.name empname
from dept d left join emp e on e.dep_id = d.id
-- Unfortunately, connecting this way, we cannot also determine the "level" of each
-- department. To do that, we would need the CONNECT BY to be reversed, i.e.,:
-- connect by prior d.parent_id = d.id
connect by d.parent_id = prior d.id
-- No "START WITH" clause
)
SELECT deptname,
count(empid) empcount,
listagg(empname,', ') within group ( order by empname) emplist
FROM intermediate
GROUP BY deptname
ORDER BY deptname;
+----------+----------+----------------------------+
| DEPTNAME | EMPCOUNT | EMPLIST |
+----------+----------+----------------------------+
| dep1 | 1 | Name3 |
| dep2 | 4 | Name1, Name2, Name4, Name5 |
| dep21 | 2 | Name1, Name5 |
| dep211 | 1 | Name5 |
| dep22 | 1 | Name2 |
+----------+----------+----------------------------+