在本章中,无涯教程将讨论PL/SQL中的集合,集合是具有相同数据类型的元素的有序组,每个元素由代表其在集合中位置的唯一下标标识。
PL/SQL提供三种集合类型-
- 索引表或关联数组 (Index-by Table Or Associative Array)
- 嵌套表(Nested Table)
- 可变大小数组(Variable-size)或Varray
无涯教程已经在" PL/SQL数组" 一章中讨论了varray。在本章中,无涯教程将讨论PL/SQL表。
索引表
index-by Table是一组键值对,每个键都是唯一的,用于定位相应的值,键可以是整数或字符串。
使用以下语法创建索引表。在这里,无涯教程将创建一个名为 table_name 的 index-by 表,该表的键将为subscript_type,而关联的值将为 element_type
TYPE type_name IS TABLE OF element_type [NOT NULL] INDEX BY subscript_type; table_name type_name;
下面的示例演示如何创建一个表来存储整数值和名称,之后再打印相同的名称列表。
DECLARE TYPE salary IS TABLE OF NUMBER INDEX BY VARCHAR2(20); salary_list salary; name VARCHAR2(20); BEGIN -- adding elements to the table salary_list('Rajnish') := 62000; salary_list('Minakshi') := 75000; salary_list('Martin') := 100000; salary_list('James') := 78000; -- printing the table name := salary_list.FIRST; WHILE name IS NOT null LOOP dbms_output.put_line ('Salary of ' || name || ' is ' || TO_CHAR(salary_list(name))); name := salary_list.NEXT(name); END LOOP; END; /
当以上代码在SQL提示符下执行时,将产生以下输出-
Salary of James is 78000 Salary of Martin is 100000 Salary of Minakshi is 75000 Salary of Rajnish is 62000 PL/SQL procedure successfully completed.
索引表的元素也可以是任何数据库表的%ROWTYPE 或任何数据库表字段的%TYPE 。无涯教程将存储在数据库中的 CUSTOMERS 表用作-
Select * from customers; +----+----------+-----+-----------+----------+ | ID | NAME | AGE | ADDRESS | SALARY | +----+----------+-----+-----------+----------+ | 1 | Ramesh | 32 | Ahmedabad | 2000.00 | | 2 | Khilan | 25 | Delhi | 1500.00 | | 3 | kaushik | 23 | Kota | 2000.00 | | 4 | Chaitali | 25 | Mumbai | 6500.00 | | 5 | Hardik | 27 | Bhopal | 8500.00 | | 6 | Komal | 22 | MP | 4500.00 | +----+----------+-----+-----------+----------+
DECLARE CURSOR c_customers is select name from customers; TYPE c_list IS TABLE of customers.Name%type INDEX BY binary_integer; name_list c_list; counter integer :=0; BEGIN FOR n IN c_customers LOOP counter := counter +1; name_list(counter) := n.name; dbms_output.put_line('Customer('||counter||'):'||name_lis t(counter)); END LOOP; END; /
当以上代码在SQL提示符下执行时,将产生以下输出-
Customer(1): Ramesh Customer(2): Khilan Customer(3): kaushik Customer(4): Chaitali Customer(5): Hardik Customer(6): Komal PL/SQL procedure successfully completed
嵌套表
嵌套表就像带有任意数量元素的一维数组,但是,嵌套表在以下方面与数组有所不同-
数组具有声明的元素数量,但嵌套表则没有,嵌套表的大小可以动态增加。
数组总是密集的,即它总是具有连续的下标。嵌套数组是密集的,但是从其中删除元素时,它可能变得稀疏。
嵌套表使用以下语法创建-
TYPE type_name IS TABLE OF element_type [NOT NULL]; table_name type_name;
此声明类似于 index-by 表的声明,但是没有 INDEX BY 子句。
嵌套表可以存储在数据库列中,在将单列表与较大的表连接在一起时,它还可用于简化SQL操作,关联数组无法存储在数据库中。
以下示例说明了嵌套表的用法-
DECLARE TYPE names_table IS TABLE OF VARCHAR2(10); TYPE grades IS TABLE OF INTEGER; names names_table; marks grades; total integer; BEGIN names := names_table('Kavita', 'Pritam', 'Ayan', 'Rishav', 'Aziz'); marks:= grades(98, 97, 78, 87, 92); total := names.count; dbms_output.put_line('Total '|| total || ' Students'); FOR i IN 1 .. total LOOP dbms_output.put_line('Student:'||names(i)||', Marks:' || marks(i)); end loop; END; /
当以上代码在SQL提示符下执行时,将产生以下输出-
Total 5 Students Student:Kavita, Marks:98 Student:Pritam, Marks:97 Student:Ayan, Marks:78 Student:Rishav, Marks:87 Student:Aziz, Marks:92 PL/SQL procedure successfully completed.
嵌套表的元素也可以是任何数据库表的%ROWTYPE 或任何数据库表字段的%TYPE,无涯教程将使用存储在数据库中的CUSTOMERS表作为-
Select * from customers; +----+----------+-----+-----------+----------+ | ID | NAME | AGE | ADDRESS | SALARY | +----+----------+-----+-----------+----------+ | 1 | Ramesh | 32 | Ahmedabad | 2000.00 | | 2 | Khilan | 25 | Delhi | 1500.00 | | 3 | kaushik | 23 | Kota | 2000.00 | | 4 | Chaitali | 25 | Mumbai | 6500.00 | | 5 | Hardik | 27 | Bhopal | 8500.00 | | 6 | Komal | 22 | MP | 4500.00 | +----+----------+-----+-----------+----------+
DECLARE CURSOR c_customers is SELECT name FROM customers; TYPE c_list IS TABLE of customerS.No.ame%type; name_list c_list := c_list(); counter integer :=0; BEGIN FOR n IN c_customers LOOP counter := counter +1; name_list.extend; name_list(counter) := n.name; dbms_output.put_line('Customer('||counter||'):'||name_list(counter)); END LOOP; END; /
当以上代码在SQL提示符下执行时,将产生以下输出-
Customer(1): Ramesh Customer(2): Khilan Customer(3): kaushik Customer(4): Chaitali Customer(5): Hardik Customer(6): Komal PL/SQL procedure successfully completed.
集合方法
PL/SQL提供了使集合更易于使用的内置集合方法,下表列出了方法及其用途-
S.No | Method Name & Purpose |
---|---|
1 |
EXISTS(n) 如果集合中的第n个元素存在,则返回TRUE;否则返回FALSE。 |
2 |
COUNT 返回集合当前包含的元素数。 |
3 |
LIMIT 检查集合的最大大小。 |
4 |
FIRST 返回使用整数下标的集合中的第一个(最小)索引号。 |
5 |
LAST 返回使用整数下标的集合中的最后一个(最大)索引号。 |
6 |
PRIOR(n) 返回集合中索引n之前的索引号。 |
7 |
NEXT(n) 返回在索引n之后的索引号。 |
8 |
EXTEND 将一个null元素追加到集合。 |
9 |
EXTEND(n) 将n个null元素追加到集合。 |
10 |
EXTEND(n,i) 将第i th 元素的 n 个副本附加到集合中。 |
11 |
TRIM 从集合的末尾删除一个元素。 |
12 |
TRIM(n) 从集合的末尾删除 n 个元素。 |
13 |
DELETE 从集合中删除所有元素,并将COUNT设置为0。 |
14 |
DELETE(n) 使用数字键或嵌套表从关联数组中删除第n个元素。如果关联数组具有字符串键,则删除与键值对应的元素。如果n为null,则DELETE(n)不执行任何操作。 |
15 |
DELETE(m,n) 从关联数组或嵌套表中删除范围为m..n的所有元素。如果m大于n或m或n为null,则DELETE(m,n)不执行任何操作。 |
集合异常
下表提供了集合异常及其引发时间-
集合异常 | 在情况中引发 |
---|---|
COLLECTION_IS_NULL | 您尝试对原子为空的集合进行操作。 |
NO_DATA_FOUND | 下标表示已删除的元素或关联数组中不存在的元素。 |
SUBSCRIPT_BEYOND_COUNT | 下标超出集合中元素的数量。 |
SUBSCRIPT_OUTSIDE_LIMIT | 下标超出允许范围。 |
VALUE_ERROR | 下标为null或不能转换为键(key)类型。如果键定义为 PLS_INTEGER 范围,并且下标超出此范围,则可能会发生此异常。 |
参考链接
https://www.learnfk.com/plsql/plsql-collections.html
标签:Customer,name,list,无涯,嵌套,Collections,SQL,集合 From: https://blog.51cto.com/u_14033984/8943733