1. 程式人生 > >Oracle:變長陣列varray 巢狀表 集合

Oracle:變長陣列varray 巢狀表 集合

 

 

oracle:變長陣列varray,巢狀表,集合

2013年08月29日 16:00:24 白天的貓頭鷹 閱讀數:1045

建立變長陣列型別

CREATE TYPE varray_type AS VARRAY(2) OF VARCHAR2(50); 

 

這個變長陣列最多可以容納兩個資料,資料的型別為 varchar2(50) 

 

更改元素型別的大小或精度

可以更改變長陣列型別和巢狀表型別 元素的大小。

ALTER TYPE varray_type 
MODIFY ELEMENT TYPE varchar2(100) CASCADE;

 

CASCADE選項吧更改傳播到資料庫中的以來物件。也可以用 INVALIDATE 選項使依賴物件無效

增加變長陣列的元素數目

ALTER TYPE vrray_name
MODIFY LIMIT 5 CASCADE;

使用變長陣列

CREATE TABLE table_name(
column_name type,
var_col_name varray_type
);

 

 獲得變長陣列的資訊

DESC[RIBE] varray_type;

SELECT * 
FROM user_varrays
WHERE type_name = varray_name;

 DESC 獲得的是 varray_type AS VARRAY(2) OF VARCHAR2(50)

填充變長陣列元素

複製程式碼

INSERT INTO table_name VALUES(
  value,
  varray_type(
    'xxxx',
    'xx',
    'x')
);

複製程式碼

 

 可以一次向變長陣列新增多個數據。

查詢變長陣列元素

SELECT *
FROM table_Name;

 

如果變長陣列中的元素有多個,會一起輸出,輸出的資料列是一個長列,跟包含物件的表一樣。 

 更改變長陣列元素

 要想更改變長陣列的一個元素,需要把其他元素一起更改,整個變長陣列作為一個整體來的。

UPDATE table_name
SET var_col_name = varray_type('xxx','xxxxxx')
WHERE expr1;

 

 

建立巢狀表型別

CREATE TYPE table_type AS TABLE OF type;

 

 其中type 可以為任何型別,包括varray 和 object ,通常object 居多。

 使用巢狀表型別

複製程式碼

CREATE TABLE table_name(
  column_name type,
  tab_col_name table_type
)
NESTED TABLE
  table_col_name
STORE AS
  next_table_name [TABLESPACE user_name];

複製程式碼

 

 建立巢狀表的時候要為巢狀表型別另外建立一個表來儲存資料, NESTED 以下的部分就是在幹這事。那個表的名稱為: next_table_name

TABLESPACE 可以將另外建立表放到另外的空間。

 獲得表資訊

SET DESCRIBE DEPTH 2
DESC[RIBE] table_name;

 

也可以直接通過資料字典來獲得巢狀表的資訊

SELECT *
FROM user_nested_tables
WHERE table_name = xxxx;

 

填充、查詢巢狀表元素

跟變長陣列方法一樣

更改巢狀表元素

跟變長陣列不同,巢狀表的元素可以單獨更改:可以插入、更改和刪除巢狀表元素。

插入

INSERT INTO TABLE(
  SELECT tab_col_name FROM table_name WHERE expr)
  VALUES(
    table_type('xxxx')
  )
);

 

 更改

複製程式碼

UPDATE TABLE(
  SELECT tab_col_name FROM table_name WHERE expr
) T
SET
  VALUE(T)  = table_type(
    'xxxx')
)
WHERE 
  VALUE(T) = table_type(
    expr2
   );

複製程式碼

 

 T 為獲得需要修改的那行資料對應的 巢狀表位置,WHERE 為判斷語句,如果table_type 為object型別,expr2 這樣寫: 'x','xx','xxx'

刪除:

複製程式碼

DELETE FROM TABLE(
  SELECT tab_col_name FROM table_name WHERE expr
)T
WHERE
  VALUE(T) = table_type(
    expr2
  );

複製程式碼

 

 

集合方法

 

EXISTS(N)

如果第n個元素存在,返回TRUE

COUNT

該函式集合元素的數目

DELETE

DELETE(n)

DELETE(n,m)

刪除集合元素

l         刪除所有元素

l         刪除第n個元素

l         刪除n到m的元素

FIRST

返回集合第一個(最小的)元素索引號,如果集合為空,返回NULL

LAST

返回集合中最後一個(最大的)元素索引號,如果集合為空,返回NULL

NEXT(n)

返回集合當前元素的下n元素的索引號,如果它不存在就返回NULL

PRIOR(n)

返回集合當前元素的前n元素的索引號,如果它不存在就返回NULL

LIMIT

返回varray中建立元素的最大個數

EXTEND

EXTEND(n)

EXTEND(n,m)

增加集合的大小。

l         新增一個,設為空

l         新增n個,設為空

l         新增n個,設為m

TRIM

TRIM(n)

從集合末尾處刪除元素

l         刪除一個

l         刪除n個

 呼叫方法是: tab_col_name.COUNT

巢狀表運算子操作

例如:

var_tab_1 table_type;

var_tab_2 table_type;

var_tab_3 table_type;

reslut BOOLEAN;

 

var_tab_1 :=table_type('1','2');

var_tab_2 :=table_type('3','4');

var_tab_3 :=table_type('2','1');

 

result:= var_tab_1 =var_tab_3  result 為true;

result:= var_tab_2 <>var_tab_3  result 為true;

 

IN 和 NOT IN 運算子

用於檢測一個巢狀表的內容是否出現在令一個巢狀表的內容中。

result:= var_tab_1 IN (var_tab_3);   result 為 TRUE;

result:= var_tab_2 NOT IN (var_tab_3);   result 為 TRUE;

 

SUBMULITSET 子集運算子

檢查一個巢狀表的內容是否為另外一個巢狀表的子集

result:= var_tab_1 SUBMULITSET OF var_tab_3  result 為TRUE;

 

MULTISET 集合運算子

返回的是一個巢狀集

MULTISET UNION

MULTISET INTERSECT

MULTISET EXCEPT

並 交  差 ,另外還有

ALL  全部

DISTINCT 去重

var_tab1 := var_tab2 MUSTISET UNION ALL var_tab2

 

CARDINALITY 獲得巢狀表中元素數目

CARDINALITY(var_tab1)

(跟count 有什麼不同。。- -!)

 

 MEMBER OF  運算子

檢測巢狀表的一個元素是否存在

'xxx' MEMBER OF var_tab1;  返回BOOLEAN 

 

SET 運算子

將傳入的巢狀表去重後返回

var_tab1 := SET (var_tab2);

 

IS A SET 

判斷時候符合每個元素都不同

result:= var_tab1 IS A SET;

 

IS EMPTY

判斷巢狀表是否為空

 

 COLLECT 運算子

 將值列表作為巢狀表返回,可以配合 CAST 運算子將返回的巢狀表強制轉換為一種巢狀表型別。

SELECT COLLECT(column_name)
FROM TABLE
WHERE expr;

 

 

POWERMULTISET 

獲得巢狀表的子巢狀表

SELECT *
FROM TABLE(
  POWERMULTISET(table_tpye('1','2'))
);

 

獲得

table_type('1')
table_type('2')
table_type('1','2')

 

注意:PS/SQL 不支援這個

 

POWERMULTISET_BY_CARDINALITY 

獲得指定長度以下的巢狀表

SELECT *
FROM TABLE(
  POWERMULTISET_BY_CARDINALITY(table_tpye('1','2'),2)
);

 

獲得

table_type('1','2')

 

 

PS/SQL 不支援這個

PL/SQL變長陣列

1.1萬

PL/SQL變長陣列時PL/SQL集合資料型別中的一種,其使用方法與PL/SQL巢狀表大同小異,唯一的區別則是變長陣列的元素的最大個數是有限制的。也即是說變長陣列的下標固定下限等於1,上限可以擴充套件。下...來自: robinson_0612

https://blog.csdn.net/zouqingfang/article/details/10526757

 

 

 

PL/SQL變長陣列

2012年03月23日 15:47:29 Leshami 閱讀數:10676 標籤: inserttableinteger擴充套件nulloracle 更多

個人分類: -----SQL/PLSQL基礎

所屬專欄: SQL PL/SQL基礎系列

版權宣告:本文為博主原創文章,歡迎擴散,擴散請務必註明出處。 https://blog.csdn.net/robinson_0612/article/details/7387647

  PL/SQL變長陣列時PL/SQL集合資料型別中的一種,其使用方法與PL/SQL巢狀表大同小異,唯一的區別則是變長陣列的元素的最大個數是有限
制的。也即是說變長陣列的下標固定下限等於1,上限可以擴充套件。下面給出具體的描述及其使用方法。

 

一、變長陣列語法
        TYPE type_name IS {VARRAY | VARYING ARRAY} (size_limit) OF  -->type_name 用於指定varray型別名,size_limit 定義varray元素的最大個數
        element_type [NOT NULL];                                    -->element_type用於指定元素的資料型別  
        varray_name TYPE_NAME;                                      -->varray_name 用於定義varray變數

 

二、變長陣列特性
          變長陣列主要的特性即是元素的最大個數是有限制
          變長陣列下標固定為1,上限可以擴充套件
          與巢狀表類似,在變長陣列宣告時自動設定為NULL值.所謂的空值指的是集合本身是空,不是針對它所擁有的元素
          故在元素引用前需要對其進行初始化

 

三、變長陣列示例

 
  1. --1、宣告變長陣列,並輸出其結果

  2. [email protected]> DECLARE

  3. 2 CURSOR name_cur IS

  4. 3 SELECT dname

  5. 4 FROM dept

  6. 5 WHERE deptno < 40;

  7. 6

  8. 7 TYPE name_type IS VARRAY( 10 ) OF dept.dname%TYPE; -->宣告一個包含10個元素的變長陣列,且且資料型別為dept.dname型別

  9. 8

  10. 9 varray_dname_tab name_type := name_type( ); -->初始化變長陣列

  11. 10 v_counter INTEGER := 0;

  12. 11 BEGIN

  13. 12 FOR name_rec IN name_cur

  14. 13 LOOP

  15. 14 v_counter :=

  16. 15 v_counter

  17. 16 + 1;

  18. 17 varray_dname_tab.EXTEND; -->使用extend進行擴充套件

  19. 18 varray_dname_tab( v_counter ) := name_rec.dname; -->按下標輸出變長陣列的所有元素

  20. 19 DBMS_OUTPUT.put_line( 'Dname ('

  21. 20 || v_counter

  22. 21 || ') is :'

  23. 22 || varray_dname_tab( v_counter ) );

  24. 23 END LOOP;

  25. 24 END;

  26. 25 /

  27. Dname (1) is :ACCOUNTING

  28. Dname (2) is :RESEARCH

  29. Dname (3) is :SALES

  30.  
  31. --2、對變長陣列直接賦初值,且使用count遍歷並輸出所有元素

  32. [email protected]> DECLARE

  33. 2 TYPE name_type IS VARRAY( 2 ) OF VARCHAR2( 10 );

  34. 3

  35. 4 varray_name_tab name_type := name_type( 'Robinson', 'Jackson' ); -->此處對varray_name_tab初始化並賦初值

  36. 5 BEGIN

  37. 6 FOR i IN 1 .. varray_name_tab.COUNT -->使用count來遍歷並輸出變長陣列的所有元素

  38. 7 LOOP

  39. 8 DBMS_OUTPUT.put_line( 'Name varray_name_tab( '

  40. 9 || i

  41. 10 || ' ) is : '

  42. 11 || varray_name_tab( i ) );

  43. 12 END LOOP;

  44. 13 END;

  45. 14 /

  46. Name varray_name_tab( 1 ) is : Robinson

  47. Name varray_name_tab( 2 ) is : Jackson

  48.  
  49. PL/SQL procedure successfully completed.

  50.  
  51. --3、超出變長陣列大小的情形

  52. [email protected]> DECLARE

  53. 2 TYPE name_type IS VARRAY( 2 ) OF VARCHAR2( 10 );

  54. 3

  55. 4 varray_name_tab name_type := name_type( 'Robinson', 'Jackson' );

  56. 5 BEGIN

  57. 6 FOR i IN 1 .. varray_name_tab.COUNT

  58. 7 LOOP

  59. 8 DBMS_OUTPUT.put_line( 'Name varray_name_tab( '

  60. 9 || i

  61. 10 || ' ) is : '

  62. 11 || varray_name_tab( i ) );

  63. 12 END LOOP;

  64. 13

  65. 14 varray_name_tab.EXTEND;

  66. 15 varray_name_tab( 3 ) := 'Johnson';

  67. 16 DBMS_OUTPUT.put_line( 'Name varray_name_tab (3) is '

  68. 17 || varray_name_tab( 3 ) );

  69. 18 END;

  70. 19 /

  71. Name varray_name_tab( 1 ) is : Robinson

  72. Name varray_name_tab( 2 ) is : Jackson

  73. DECLARE

  74. *

  75. ERROR at line 1:

  76. ORA-06532: Subscript outside of limit

  77. ORA-06512: at line 14

  78.  
  79. --4、儲存變長陣列到資料庫及修改變長陣列

  80. [email protected]> CREATE OR REPLACE TYPE varray_phone IS VARRAY( 2 ) OF VARCHAR2( 40 ); -->建立變長陣列型別

  81. 2 /

  82.  
  83. Type created.

  84.  
  85. [email protected]> CREATE TABLE tb_emp -->建立表tb_emp且其中一列使用到了變長陣列

  86. 2 (

  87. 3 empno NUMBER( 4 )

  88. 4 , ename VARCHAR2( 10 )

  89. 5 , phone varray_phone -->列phone使用到了變長陣列

  90. 6 )

  91. 7 ;

  92.  
  93. Table created.

  94.  
  95. --插入新記錄到變長陣列

  96. [email protected]> insert into tb_emp select 6666,'Robinson',varray_phone('13423456789','075520123650') from dual;

  97.  
  98. 1 row created.

  99.  
  100. [email protected]> insert into tb_emp select 7777,'Jackson',varray_phone('13423456789','075520123650') from dual;

  101.  
  102. 1 row created.

  103.  
  104. [email protected]> commit;

  105.  
  106. Commit complete.

  107.  
  108. [email protected]> col phone format a25

  109. [email protected]> select * from tb_emp; -->檢視插入的記錄

  110.  
  111. EMPNO ENAME PHONE

  112. ---------- ---------- -------------------------

  113. 6666 Robinson VARRAY_PHONE('13423456789

  114. ', '075520123650')

  115.  
  116. 7777 Jackson VARRAY_PHONE('13423456789

  117. ', '075520123650')

  118.  
  119. -->插入變長陣列包含3個元素的記錄,此時提示超出了變長陣列範圍

  120. [email protected]> insert into tb_emp select 8888,'Johnson',varray_phone('13423456789','075520123650','010123')

  121. 2 from dual;

  122. insert into tb_emp select 8888,'Johnson',varray_phone('13423456789','075520123650','010123')

  123. *

  124. ERROR at line 1:

  125. ORA-22909: exceeded maximum VARRAY limit

  126.  
  127. -->修改變長陣列長度範圍限制到3,且使用cascade,即及聯修改到表tb_emp

  128. [email protected]> alter type varray_phone modify limit 3 cascade;

  129.  
  130. Type altered.

  131.  
  132. -->再次插入資料成功

  133. [email protected]> insert into tb_emp select 8888,'Johnson',varray_phone('13423456789','075520123650','010123')

  134. 2 from dual;

  135.  
  136. 1 row created.

  137.  
  138. -->插入變長陣列元素超出預定義字串長度時收到錯誤提示

  139. [email protected]> insert into tb_emp select 8888,'Johnson',varray_phone('13423456789000000000000000000000000000000000000')

  140. 2 from dual;

  141. insert into tb_emp select 8888,'Johnson',varray_phone('13423456789000000000000000000000000000000000000')

  142. *

  143. ERROR at line 1:

  144. ORA-22814: attribute or element value is larger than specified in type

  145.  
  146. -->修改變長陣列元素的資料長度到varchar2(60),且使用cascade,即及聯修改到表tb_emp

  147. [email protected]> alter type varray_phone modify element type varchar2(60) cascade;

  148.  
  149. Type altered.

  150.  
  151. -->再次插入資料成功

  152. [email protected]> insert into tb_emp select 8888,'Johnson',varray_phone('13423456789000000000000000000000000000000000000')

  153. 2 from dual;

  154.  
  155. 1 row created.

  156.  
  157. -->檢視變長陣列的定義

  158. [email protected]> select parent_table_name,parent_table_column,type_name from user_varrays;

  159.  
  160. PARENT_TABLE_NA PARENT_TABLE_COLUMN TYPE_NAME

  161. --------------- ------------------------------ ------------------------------

  162. TB_EMP PHONE VARRAY_PHONE

四、更多參考

啟用使用者程序跟蹤

父遊標、子游標及共享遊標

繫結變數及其優缺點

dbms_xplan之display_cursor函式的使用

dbms_xplan之display函式的使用

執行計劃中各欄位各模組描述

使用 EXPLAIN PLAN 獲取SQL語句執行計劃

啟用 AUTOTRACE 功能

函式使得索引列失效

Oracle 繫結變數窺探

PL/SQL 聯合陣列與巢狀表

https://blog.csdn.net/leshami/article/details/7387647