1. 程式人生 > >Oracle CONNECT BY的用法

Oracle CONNECT BY的用法

 Oracle中可以通過START WITH . . . CONNECT BY . . .子句來實現SQL的層次查詢.
自從Oracle 9i開始,可以通過 SYS_CONNECT_BY_PATH 函式實現將父節點到當前行內容以“path”或者層次元素列表的形式顯示出來。

自從Oracle 10g 中,還有其他更多關於層次查詢的新特性 。例如,有的時候使用者更關心的是每個層次分支中等級最低的內容。
那麼你就可以利用偽列函式CONNECT_BY_ISLEAF來判斷當前行是不是葉子。如果是葉子就會在偽列中顯示“1”,
如果不是葉子而是一個分支(例如當前內容是其他行的父親)就顯示“0”。

在Oracle 10g 之前的版本中,如果在你的樹中出現了環狀迴圈(如一個孩子節點引用一個父親節點),
Oracle 就會報出一個錯誤提示:“ ORA-01436: CONNECT BY loop in user data”。如果不刪掉對父親的引用就無法執行查詢操作。
而在 Oracle 10g 中,只要指定“NOCYCLE

”就可以進行任意的查詢操作。與這個關鍵字相關的還有一個偽列——CONNECT_BY_ISCYCLE
如果在當前行中引用了某個父親節點的內容並在樹中出現了迴圈,那麼該行的偽列中就會顯示“1”,否則就顯示“0”。

The start with .. connect by clause can be used to select data that has a hierarchical relationship
(usually some sort of parent->child, boss->employee or thing->parts).
It is also being used when an sql execution plan is explained.

syntax:
select ... [start with initial-condition] connect by [nocycle] recurse-condition

level
With level it is possible to show the level in the hierarchical relation of all the data.

--oracle 9i
sys_connect_by_path
With sys_connect_by_path it is possible to show the entire path from the top level down to the 'actual' child.

--oracle 10g
connect_by_root
connect_by_root is a new operator that comes with Oracle 10g and enhances the ability to perform hierarchical queries.
connect_by_is_leaf
connect_by_isleaf is a new operator that comes with Oracle 10g and enhances the ability to perform hierarchical queries.
connect_by_iscycle
connect_by_is_cycle is a new operator that comes with Oracle 10g and enhances the ability to perform hierarchical queries.

下面舉例說明:
[例1]
建立一個部門表,這個表有4個欄位,分別對應部門ID,部門名稱,上級部門ID ,本部門直屬員工人數。
create table DEP

DEPID      number(10) not null, 
DEPNAME    varchar2(32), 
UPPERDEPID number(10),
NUMOFEMP   number(10)
) ;
下面插入一些資料,結果如下:

[email protected]> select * from dep;

     DEPID DEPNAME         UPPERDEPID   NUMOFEMP
---------- --------------- ---------- ----------
         0 Dev Center                          2
         1 DevA                     0          3
         2 DevA Team1               1         10
         3 DevA Team2               1          8
         4 DevB                     0          2
         5 DevB Team1               4         12
         6 Test Center                         1
         7 Test Team1               6          5
         8 Test Team2               6          5

9 rows selected.

現在根據“CONNECT BY”來實現樹狀查詢:

column depname format a15;
column rootdep format a15;
column path format a30;

select rpad(' ', 2*(level-1), '-') || depname "DEPNAME",
connect_by_root depname "ROOTDEP",
connect_by_isleaf "ISLEAF",
level,
sys_connect_by_path(depname, '/') "PATH"
from dep
start with upperdepid is null
connect by prior depid = upperdepid
/

結果如下:

DEPNAME         ROOTDEP             ISLEAF      LEVEL PATH
--------------- --------------- ---------- ---------- ---------------------------
Dev Center      Dev Center               0          1 /Dev Center
 -DevA          Dev Center               0          2 /Dev Center/DevA
 ---DevA Team1  Dev Center               1          3 /Dev Center/DevA/DevA Team1
 ---DevA Team2  Dev Center               1          3 /Dev Center/DevA/DevA Team2
 -DevB          Dev Center               0          2 /Dev Center/DevB
 ---DevB Team1  Dev Center               1          3 /Dev Center/DevB/DevB Team1
Test Center     Test Center              0          1 /Test Center
 -Test Team1    Test Center              1          2 /Test Center/Test Team1
 -Test Team2    Test Center              1          2 /Test Center/Test Team2


下面計算Dev Center和Test Center部門的總人數:

select ROOTDEPID, sum(numofemp) "TOTALEMP"
from (select connect_by_root depid "ROOTDEPID", numofemp from dep
start with upperdepid is null
connect by prior depid = upperdepid)
group by ROOTDEPID
/

結果如下:

 ROOTDEPID   TOTALEMP
---------- ----------
         6         11
         0         37


[例2]
通過CONNECT BY用於十六進度轉換為十進位制

CREATE OR REPLACE FUNCTION f_hex_to_dec(p_str IN VARCHAR2) RETURN VARCHAR2 IS
    ----------------------------------------------------------------------------------------------------------------------
    -- 物件名稱: f_hex_to_dec
    -- 物件描述: 十六進位制轉換十進位制
    -- 輸入引數: p_str 十六進位制字串
    -- 返回結果: 十進位制字串
    -- 測試用例: SELECT f_hex_to_dec('78A') FROM dual;
    ----------------------------------------------------------------------------------------------------------------------
    v_return  VARCHAR2(4000);
  BEGIN
    SELECT SUM(DATA) INTO v_return
      FROM (SELECT (CASE upper(substr(p_str, rownum, 1))
                     WHEN 'A' THEN '10'
                     WHEN 'B' THEN '11'
                     WHEN 'C' THEN '12'
                     WHEN 'D' THEN '13'
                     WHEN 'E' THEN '14'
                     WHEN 'F' THEN '15'
                     ELSE substr(p_str, rownum, 1)
                   END) * power(16, length(p_str) - rownum) DATA
              FROM dual
            CONNECT BY rownum <= length(p_str));
    RETURN v_return;
  EXCEPTION
    WHEN OTHERS THEN
      RETURN NULL;
  END;


說明:
1. CONNECT BY rownum <= length(p_str))對輸入的字串進行逐個遍歷
2. 通過CASE語句,來解析十六進位制中的A-F對應的10進位制值

測試結果如下:

[email protected]> variable dec varchar2(32);
[email protected]> exec :dec := f_hex_to_dec('1FF');

PL/SQL procedure successfully completed.

[email protected]> print dec

DEC
----------
511


[例3]
通過CONNECT BY生成序列
對於connect by,現在大多數人已經很熟悉了,connect by中的條件就表示了父子之間的連線關係,比如 connect by id=prior pid。

但如果connect by中的條件沒有表示記錄之間的父子關係
那會出現什麼情況?
常見的,connect by會在構造序列的時候使用
用select rownum from dual connect by rownum<xxx 代替早期版本的 select rownum from all_objects where rownum <xxx

我們注意到,dual是一個只有一條記錄的表,如果表有多條記錄,將會怎樣?
下面開始實驗
CREATE TABLE T(ID VARCHAR2(1 BYTE));

INSERT INTO T ( ID ) VALUES ( 'A');
INSERT INTO T ( ID ) VALUES ( 'B');
INSERT INTO T ( ID ) VALUES ( 'C');
COMMIT;
然後執行以下查詢:

[email protected]> column id format a2;
[email protected]> select id,level from t connect by level<2;

ID      LEVEL
-- ----------
A           1
B           1
C           1

[email protected]> select id,level from t connect by level<3;


ID      LEVEL
-- ----------
A           1
A           2
B           2
C           2
B           1
A           2
B           2
C           2
C           1
A           2
B           2
C           2

12 rows selected.


無需多說,我們很快可以找到其中的規律,假設表中有N條記錄
則記F(N,l)為 select id,level from t connect by level<l 的結果集數目
那麼,
F(N,1)=N
F(N,l) = F(N,l-1)*N+N

於是可以總結出
F(N,l)=∑power(N,p), p取值為[1,l)

要解釋,也很容易。
當連線條件不能限制記錄之間的關係時
每一條記錄都可以作為自己或者其他記錄的葉子
如下所示:
A          1
A          2
A          3
B          3
C          3
B          2
A          3
B          3
C          3
C          2
A          3
B          3
C          3


在這裡,我們看到的是
Oracle採用了深度優先的演算法

轉載自:
http://www.blogjava.net/wxqxs/archive/2008/08/15/222338.html
http://www.cnblogs.com/lettoo/archive/2010/08/03/1791239.html
http://www.itpub.net/thread-994465-1-1.html

相關推薦

oracle connect by用法以及with遞迴

環境:11g 準備: 在oracle中start with connect by (prior) 用來對樹形結構的資料進行查詢。其中start with conditon 給出的是資料搜尋範圍, connect by後面給出了遞迴查詢的條件; 涉及的偽列及函式:

oracle connect by用法

1、基本語法 select * from table [start with condition1] connect by [prior] id=parentid 一般用來查詢存在父子關係的資料,也就是樹形結構的資料;其返還的資料也能夠明確的區分

Oracle層級詢語句connect by 用法詳解

condition 選擇 sta sele 2.3 循環 條件定義 nbsp 關系 如果表中包含層級數據,那麽你就可以使用層級查詢從句選擇行層級順序。 1.層級查詢從句語法 層級查詢從句語法: { CONNECT BY [ NOCYCLE ] condition [AND

oracle樹操作查詢start with和connect by用法和理解

oracle樹操作查詢start with和connect by用法和理解 connect by 是結構化查詢中用到的,其基本語法是 select … from tablename start with 條件1 connect by 條件2 where 條件3; 例: select

Oracle connect by prior用法

Oracle 中 connect by prior的用法:層級查詢,用於查詢具有層級關係(樹狀結構)的資料(父子關係、僱傭關係)。 語法: select [level]  ... from table_name start with condition1 connect

oracleConnect By用法介紹

為解決oracle中自連線查詢不適合操作大表的情況,採用connect by 方式實現。oracle中可以用START WITH...CONNECT BY PRIOR子句實現遞迴查詢,connect by 在結構化查詢中應用。 基本語法:             selec

Oracle層級查詢語句(hierarchical query)connect by 用法詳解

摘要: 1、connect by中的條件就表示了父子之間的連線關係,比如 connect by id=prior  pid。 2、prior,表示在一表上,prior所在列pid的某一值A的記錄行的父親是列id上值等於列pid上值A的記錄行。 3、LEVEL偽列

Oracle CONNECT BY用法

 Oracle中可以通過START WITH . . . CONNECT BY . . .子句來實現SQL的層次查詢. 自從Oracle 9i開始,可以通過 SYS_CONNECT_BY_PATH 函式實現將父節點到當前行內容以“path”或者層次元素列表的形式顯示出來。

Oracle層級詢語句connect by 用法詳解

SQL> select e.empno, e.ename, e.mgr, level, connect_by_iscycle "CYCLE"   2    from emp e   3  connect by nocycle prior e.empno = e.mgr   4   start with

oracle connect by從句中的rownum

在構造資料時,對於單行的基本資料來說,connect by level與connect by rownum可以說是等效的。比如說下面的兩個語句的結果集是一樣的。 with tab1 as ( select 'a' id, 2 num from dual ) select*from t

oracle connect by從句中的

在構造資料時,對於單行的基本資料來說,connect by level與connect by rownum可以說是等效的。比如說下面的兩個語句的結果集是一樣的。 with tab1 as ( select 'a' id, 2 num from dual ) se

Oracle connect by level 小結(一)

1. 準備測試用表 CREATE TABLE T (ID VARCHAR2(10)); INSERT INTO T VALUES ('A'); INSERT INTO T VALUES ('B'); INSERT INTO T VALUES ('C');

Oracle connect by的使用

原文地址:http://blog.csdn.net/wang_yunj/article/details/51040029 有時候會遇到顯示展示每個時段(比如需要統計1天的24個小時的業務量以及交易金額等)的統計資訊,沒有交易的時段也要展示出來,但sql語句查詢出的結果分

ORACLE--Connect By、Level、Start With的使用(Hierarchical query-層次查詢)

查詢員工編號為7369的領導: 1 SELECT LEVEL,E.* FROM EMP E CONNECT BY PRIOR E.MGR = E.EMPNO START WITH E.EMPNO = 7876 2 ORDER BY LEVEL DESC "start with" -- this id

Oracle Connect By Prior(遞迴查詢)簡潔例項

oracle中的select語句可以用START WITH...CONNECT BY PRIOR子句實現遞迴查詢,connect by 是結構化查詢中用到的,其基本語法是:select ... fro

Oracle高級函數篇之遞歸查詢start with connect by prior簡單用法

上一條 prior 結果 表結構 from oracl 結構 ... 可能 路飛:“ 把原來CSDN的博客轉移到博客園咯!” 前段時間,自己負責的任務中剛好涉及到了組織關系的業務需求,自己用了oracle遞歸查詢。下面簡單來舉個例子。在工作中我們經常會遇到有一定組織關系層次

oracle中 start with .. connect by prior.. 用法簡介

bsp 簡單 parent 目錄樹 lena prior conn 就是 比較 我們經常會將一個比較復雜的目錄樹存儲到一個表中。或者將一些部門存儲到一個表中,而這些部門互相有隸屬關系。這個時候你就會用到connect by prior start with。oracle 提

Oracle樹查詢,start with connect by prior 遞迴查詢用法(轉載)

本人覺得這個寫的真不錯,實用性強,就轉載過來了 這個子句主要是用於B樹結構型別的資料遞迴查詢,給出B樹結構型別中的任意一個結點,遍歷其最終父結點或者子結點。 先看原始資料: 1 create table a_test 2 ( parentid varchar2(10), 3

oracle中start with和connect by用法理解

 1.樹結構的描述     樹結構的資料存放在表中,資料之間的層次關係即父子關係,通過表中的列與列間的關係來描述,如EMP表中的EMPNO和MGR,EMPNO表示該僱員的編號,MGR表示領導該僱員的人的編號,即子節點的MGR值等於父節點的EMPNO值。在表的每一行中都有一個表示父節點的MGR(除根節點外),通

Oracle plsql遞迴統計所有節點下的子節點&內容個數——start with connect by prior用法

有些情況下,在存有層級關係的表中,id欄位值本身會包含層級關係,例如每兩位表示一級。 要想只統計某個或某幾個層級節點下子節點+內容個數,就需要使用類似like ‘01%’的過濾條件即可。 例如select count(1) from test_category t wh