Преобразование родительских дочерних строк в JSON в Oracle - PullRequest
0 голосов
/ 27 ноября 2018

Есть ли способ создать объект JSON в Oracle для данных родительских и дочерних отношений?Например организационная структура.Таблица содержит

EmpId   Name    Title   ManagerId
1       John    GM          0
2       Smith   Manager     1
3       Jason   Manager     1
4       Will    IP1         3
5       Jade    AM          3
6       Mark    IP2         5
7       Jane    AM2         5
8       Tamara  M1          1
9       Dory    M2          1

Ожидается нечто вроде ниже объекта JSON.

 {
      'name': 'John',
      'title': 'GM',
      'children': [
        { 'name': 'Smith', 'title': 'manager' },
        { 'name': 'Jason', 'title': 'manager',
          'children': [
            { 'name': 'Will', 'title': 'IP1' },
            { 'name': 'Jade', 'title': 'AM',
              'children': [
                { 'name': 'Mark', 'title': 'IP2' },
                { 'name': 'Jane', 'title': 'AM2' }
              ]
            }
          ]
        },            
        { 'name': 'Tamara', 'title': 'M1' },
        { 'name': 'Dory', 'title': 'M2' }
      ]
    }

1 Ответ

0 голосов
/ 27 ноября 2018

Oracle Database 12.2 имеет ряд функций генерации JSON.Но они имеют ограниченное использование.Вам нужно создать документ рекурсивно.

Я полагаю, что это требует немного ручной работы.

Сначала используйте рекурсивный запрос для создания организационной диаграммы, добавив, какой уровень находится в иерархии у каждого человека.

Затем создайте JSON следующим образом:

  • Если уровень для следующей строки превышает текущий, сотрудник является менеджером.И вам нужно запустить дочерний массив.В противном случае вернуть объект JSON для текущей строки
  • Если текущая строка является последней в дереве, вам необходимо закрыть N массивов и объектов.N - это глубина строки в дереве минус один.
  • В противном случае, если следующий ряд находится на более низком уровне, чем текущий, вам необходимо закрыть (текущий уровень - следующий уровень) массивы и объекты
  • Тогда, если следующий уровень равен или меньше текущего, добавьте запятую

, которая дает что-то вроде:

create table t (
  EmpId   int,
  Name    varchar2(10),
  Title   varchar2(10),
  ManagerId int
);

insert into t values (1, 'John', 'GM' , 0 );
insert into t values (2, 'Smith', 'Manager' , 1 );
insert into t values (3, 'Jason', 'Manager' , 1 );
insert into t values (4, 'Will', 'IP1' , 3 );
insert into t values (5, 'Jade', 'AM' , 3 );
insert into t values (6, 'Mark', 'IP2' , 5 );
insert into t values (7, 'Jane', 'AM2' , 5 );
insert into t values (8, 'Tamar', 'M1' , 1 );
insert into t values (9, 'Dory', 'M2' , 1 );
commit;

with chart ( 
  empid, managerid, name, title, lvl
) as (
  select empid, managerid, 
         name, title, 1 lvl
  from   t
  where  empid = 1
  union all
  select t.empid, t.managerid, 
         t.name, t.title,
         lvl + 1 lvl
  from   chart c
  join   t
  on     c.empid = t.managerid
) search depth first by empid set seq,
  jdata as (
  select case
           /* The employee has reports */
           when lead ( lvl ) over ( order by seq ) > lvl then
             '{"name": "' || name ||
             '", "title": "' || title ||
             '", "children": ['
           else
             json_object ( 'name' value name, 'title' value title )
         end ||
         case
           /* Close arrays & objects */
           when lead ( lvl ) over ( order by seq ) is null then
             lpad ( ']}', ( lvl - 1 ) * 2, ']}' ) 
           when lead ( lvl ) over ( order by seq ) < lvl then
             lpad ( ']}', ( lvl - lead ( lvl ) over ( order by seq ) ) * 2, ']}' )
         end ||
         case
           /* Add closing commas */
           when lead ( lvl ) over ( order by seq ) <= lvl then
             ','
         end j,
         lead ( lvl ) over ( order by seq ) nlvl,
         seq, lvl
  from   chart
)
  select json_query ( 
           listagg ( j ) 
             within group ( order by seq ), 
           '$' returning varchar2 pretty
         ) chart_json
  from   jdata;

CHART_JSON                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                 
{
  "name" : "John",
  "title" : "GM",
  "children" :
  [
    {
      "name" : "Smith",
      "title" : "Manager"
    },
    {
      "name" : "Jason",
      "title" : "Manager",
      "children" :
      [
        {
          "name" : "Will",
          "title" : "IP1"
        },
        {
          "name" : "Jade",
          "title" : "AM",
          "children" :
          [
            {
              "name" : "Mark",
              "title" : "IP2"
            },
            {
              "name" : "Jane",
              "title" : "AM2"
            }
          ]
        }
      ]
    },
    {
      "name" : "Tamar",
      "title" : "M1"
    },
    {
      "name" : "Dory",
      "title" : "M2"
    }
  ]
}  
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...