Подсчет иерархии SQL-запросов Oracle - PullRequest
0 голосов
/ 08 июня 2018

У меня есть 2 таблицы 1- отделы (некоторые отделы входят в состав других отделов).
2- сотрудники, которые работают в отделах

WITH `d1` AS (
      SELECT 1 ID, 'dep1' NAME,    null Parent_id UNION ALL
      SELECT 2, 'dep2',           null             UNION ALL
      SELECT 3, 'dep21',  2                UNION ALL
      SELECT 4,'dep22',                 2 
    )
WITH `d2` AS (
  SELECT 1 ID, 'Name1' NAME,    3 DEP_id UNION ALL
  SELECT 2,         'Name2',    4             UNION ALL
  SELECT 3,         'Name3',     1               UNION ALL
  SELECT 4,         'Name4',    2             UNION ALL

Мне нужно найти количество сотрудников в каждомотдел, в том числе родительский.Я думаю, что я должен использовать функцию «соединить через», но я не знаю, как я могу ее использовать.Результат:

ID  Qty
1   1
2   3
3   1
4   1

1 Ответ

0 голосов
/ 08 июня 2018

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                      |
+----------+----------+----------------------------+
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...