PL/SQL - 集合


在本章中,我们将讨论 PL/SQL 中的集合。集合是具有相同数据类型的有序元素组。每个元素都由唯一的下标标识,该下标表示其在集合中的位置。

PL/SQL 提供三种集合类型 -

  • 按表或关联数组索引
  • 嵌套表
  • 可变大小数组或 Varray

Oracle 文档为每种类型的集合提供了以下特征 -

收藏类型 元件数量 下标类型 密集或稀疏 创建地点 可以是对象类型属性
关联数组(或索引表) 无界 字符串或整数 任何一个 仅在 PL/SQL 块中
嵌套表 无界 整数 开始密集,可能变得稀疏 在 PL/SQL 块中或在架构级别 是的
可变大小数组(Varray) 有界 整数 始终浓密 在 PL/SQL 块中或在架构级别 是的

我们已经在“PL/SQL 数组”一章中讨论了 varray 。在本章中,我们将讨论 PL/SQL 表。

两种类型的 PL/SQL 表(即索引表和嵌套表)具有相同的结构,并且使用下标表示法来访问它们的行。然而,这两种类型的表在一方面有所不同;嵌套表可以存储在数据库列中,而索引表则不能。

按表索引

索引(也称为关联数组)是一组键值对。每个键都是唯一的,用于定位对应的值。键可以是整数或字符串。

使用以下语法创建索引表。在这里,我们创建一个名为table_name的索引表,其键为 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子句。

嵌套表可以存储在数据库列中。它还可以用于简化将单列表与更大的表连接起来的 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 提供了内置的集合方法,使集合更易于使用。下表列出了这些方法及其目的 -

序列号 方法名称和目的
1

存在(n)

如果集合中第 n 个元素存在,则返回 TRUE;否则返回 FALSE。

2

数数

返回集合当前包含的元素数。

3

限制

检查集合的最大大小。

4

第一的

返回使用整数下标的集合中的第一个(最小)索引号。

5

最后的

返回使用整数下标的集合中最后一个(最大)索引号。

6

先前(n)

返回集合中索引 n 之前的索引号。

7

下一个(n)

返回索引 n 之后的索引号。

8

延长

将一个空元素追加到集合中。

9

延长(n)

将 n 个 null 元素追加到集合中。

10

扩展(n,i)

将第 i个元素的n 个副本追加到集合中。

11

修剪

从集合末尾移除一个元素。

12

修剪(n)

从集合末尾删除n 个元素。

13

删除

从集合中删除所有元素,将 COUNT 设置为 0。

14

删除(n)

从具有数字键或嵌套表的关联数组中删除第n个元素。如果关联数组有字符串键,则删除该键值对应的元素。如果n为空,则DELETE(n)不执行任何操作。

15

删除(m,n)

从关联数组或嵌套表中删除m..n范围内的所有元素。如果m大于n或者mn为空,则 DELETE(m,n)不执行任何操作。

集合例外

下表提供了集合异常以及引发异常的时间 -

集合异常 成长环境
COLLECTION_IS_NULL 您尝试对Atomics空集合进行操作。
没有找到数据 下标表示已删除的元素,或关联数组中不存在的元素。
SUBSCRIPT_BEYOND_COUNT 个 下标超过集合中元素的数量。
SUBSCRIPT_OUTSIDE_LIMIT 下标超出了允许的范围。
VALUE_ERROR 下标为空或不可转换为键类型。如果键定义为PLS_INTEGER范围,并且下标超出此范围,则可能会发生此异常。