0

SQL ではかなり一般的な階層クエリを実行する必要があります。しかし、Xquery を使用して xml ドキュメントに対して同じことを行う必要があります。

たとえば、再帰的な CONNECT BY 機能に関する標準の Oracle チュートリアルをモデル化しています。これはテーブルですが、xml に変換されています。

<EMPLOYEES>
  <EMP>
    <EMPNO>7369</EMPNO>
    <ENAME>SMITH</ENAME>
    <JOB>CLERK</JOB>
    <MGR>7902</MGR>
    <HIREDATE>17-DEC-80</HIREDATE>
    <SAL>800</SAL>
  </EMP>
  <EMP>
    <EMPNO>7499</EMPNO>
    <ENAME>ALLEN</ENAME>
    <JOB>SALESMAN</JOB>
    <MGR>7698</MGR>
    <HIREDATE>20-FEB-81</HIREDATE>
    <SAL>1600</SAL>
    <COMM>300</COMM>
  </EMP>
  <EMP>
    <EMPNO>7521</EMPNO>
    <ENAME>WARD</ENAME>
    <JOB>SALESMAN</JOB>
    <MGR>7698</MGR>
    <HIREDATE>22-FEB-81</HIREDATE>
    <SAL>1250</SAL>
    <COMM>500</COMM>
  </EMP>
  <EMP>
    <EMPNO>7566</EMPNO>
    <ENAME>JONES</ENAME>
    <JOB>MANAGER</JOB>
    <MGR>7839</MGR>
    <HIREDATE>02-APR-81</HIREDATE>
    <SAL>2975</SAL>
  </EMP>
  <EMP>
    <EMPNO>7654</EMPNO>
    <ENAME>MARTIN</ENAME>
    <JOB>SALESMAN</JOB>
    <MGR>7698</MGR>
    <HIREDATE>28-SEP-81</HIREDATE>
    <SAL>1250</SAL>
    <COMM>1400</COMM>
  </EMP>
  <EMP>
    <EMPNO>7698</EMPNO>
    <ENAME>BLAKE</ENAME>
    <JOB>MANAGER</JOB>
    <MGR>7839</MGR>
    <HIREDATE>01-MAY-81</HIREDATE>
    <SAL>2850</SAL>
  </EMP>
  <EMP>
    <EMPNO>7782</EMPNO>
    <ENAME>CLARK</ENAME>
    <JOB>MANAGER</JOB>
    <MGR>7839</MGR>
    <HIREDATE>09-JUN-81</HIREDATE>
    <SAL>2450</SAL>
  </EMP>
  <EMP>
    <EMPNO>7788</EMPNO>
    <ENAME>SCOTT</ENAME>
    <JOB>ANALYST</JOB>
    <MGR>7566</MGR>
    <HIREDATE>19-APR-87</HIREDATE>
    <SAL>3000</SAL>
  </EMP>
  <EMP>
    <EMPNO>7839</EMPNO>
    <ENAME>KING</ENAME>
    <JOB>PRESIDENT</JOB>
    <HIREDATE>17-NOV-81</HIREDATE>
    <SAL>5000</SAL>
  </EMP>
  <EMP>
    <EMPNO>7844</EMPNO>
    <ENAME>TURNER</ENAME>
    <JOB>SALESMAN</JOB>
    <MGR>7698</MGR>
    <HIREDATE>08-SEP-81</HIREDATE>
    <SAL>1500</SAL>
    <COMM>0</COMM>
  </EMP>
  <EMP>
    <EMPNO>7876</EMPNO>
    <ENAME>ADAMS</ENAME>
    <JOB>CLERK</JOB>
    <MGR>7788</MGR>
    <HIREDATE>23-MAY-87</HIREDATE>
    <SAL>1100</SAL>
  </EMP>
  <EMP>
    <EMPNO>7900</EMPNO>
    <ENAME>JAMES</ENAME>
    <JOB>CLERK</JOB>
    <MGR>7698</MGR>
    <HIREDATE>03-DEC-81</HIREDATE>
    <SAL>950</SAL>
  </EMP>
  <EMP>
    <EMPNO>7902</EMPNO>
    <ENAME>FORD</ENAME>
    <JOB>ANALYST</JOB>
    <MGR>7566</MGR>
    <HIREDATE>03-DEC-81</HIREDATE>
    <SAL>3000</SAL>
  </EMP>
  <EMP>
    <EMPNO>7934</EMPNO>
    <ENAME>MILLER</ENAME>
    <JOB>CLERK</JOB>
    <MGR>7782</MGR>
    <HIREDATE>23-JAN-82</HIREDATE>
    <SAL>1300</SAL>
  </EMP>
</EMPLOYEES>

SYS_CONNECT_BY_PATH 関数をエミュレートし、特定の従業員の組織階層を作成しようとしています。たとえば、従業員ADAMSの場合...

キング・ジョーンズ・スコット・アダムス

リンクは従業員 ID と彼のマネージャー コードの間にあります。

再帰的な文字列連結関数がそのトリックを行うように見えますが、私は Xquery に慣れていないため、構文を機能させることができません。

私の混乱の一部は、「ヒエラルキー」の概念に由来しています。これは、私が追跡する必要がある文書階層ではありません。階層はデータにあります。リンクされたリストのようにレンダリングされます。

任意の考えをいただければ幸いです。

4

1 に答える 1

0

理想的には、XML は階層的にフォーマットされるため、組織は XML で暗黙的に示され、従業員の祖先 (または子孫) を取得するだけで済みます。この場合、従業員からマネージャー - 従業員への ID ベースのリンクのパスをたどる再帰関数を作成できます。

declare function local:boss-R(
  $emp as element(EMP)?
) as xs:string*
{
  if (empty($emp)) then ()
  else 
    let $boss := $emp/ancestor::EMPLOYEES/EMP[EMPNO eq $emp/MGR]
    return ($boss/ENAME/string(), local:boss-R($boss))
};

declare function local:org(
  $emp as element(EMP)
) as xs:string+
{
  $emp/ENAME, local:boss-R($emp)
};

let $emp := doc('employees.xml')/EMPLOYEES/EMP[1]
return reverse(local:org($emp))
于 2013-11-08T16:33:56.440 に答える