- PL/SQL 教程
- PL/SQL - 首页
- PL/SQL - 概述
- PL/SQL - 环境
- PL/SQL - 基本语法
- PL/SQL - 数据类型
- PL/SQL - 变量
- PL/SQL - 常量和字面量
- PL/SQL - 运算符
- PL/SQL - 条件语句
- PL/SQL - 循环语句
- PL/SQL - 字符串
- PL/SQL - 数组
- PL/SQL - 存储过程
- PL/SQL - 函数
- PL/SQL - 游标
- PL/SQL - 记录
- PL/SQL - 异常处理
- PL/SQL - 触发器
- PL/SQL - 包
- PL/SQL - 集合
- PL/SQL - 事务
- PL/SQL - 日期和时间
- PL/SQL - DBMS 输出
- PL/SQL - 面向对象
- PL/SQL 有用资源
- PL/SQL - 问答
- PL/SQL - 快速指南
- PL/SQL - 有用资源
- PL/SQL - 讨论
PL/SQL - 集合
本章将讨论 PL/SQL 中的集合。集合是具有相同数据类型的一组有序元素。每个元素都由一个唯一的下标标识,该下标表示其在集合中的位置。
PL/SQL 提供三种集合类型:
- 按索引的表或关联数组
- 嵌套表
- 可变大小数组或 VARRAY
Oracle 文档为每种类型的集合提供了以下特性:
| 集合类型 | 元素数量 | 下标类型 | 稠密或稀疏 | 创建位置 | 可以是对象类型属性 |
|---|---|---|---|---|---|
| 关联数组(或按索引的表) | 无界 | 字符串或整数 | 任意一种 | 仅在 PL/SQL 块中 | 否 |
| 嵌套表 | 无界 | 整数 | 初始稠密,可以变为稀疏 | 在 PL/SQL 块中或在模式级别 | 是 |
| 可变大小数组 (VARRAY) | 有界 | 整数 | 始终稠密 | 在 PL/SQL 块中或在模式级别 | 是 |
我们已经在'PL/SQL 数组'章节中讨论了 VARRAY。本章将讨论 PL/SQL 表。
两种 PL/SQL 表(即按索引的表和嵌套表)具有相同的结构,并且它们的行的访问使用下标表示法。但是,这两种类型的表在一个方面有所不同;嵌套表可以存储在数据库列中,而按索引的表则不能。
按索引的表
按索引的表(也称为关联数组)是一组键值对。每个键都是唯一的,用于查找相应的 value。键可以是整数或字符串。
按索引的表使用以下语法创建。这里,我们正在创建一个名为table_name的按索引的表,其键将是`subscript_type`类型,关联的值将是`element_type`类型
TYPE type_name IS TABLE OF element_type [NOT NULL] INDEX BY subscript_type; table_name type_name;
示例
以下示例显示了如何创建一个表来存储整数 value 以及名称,然后打印相同的名称列表。
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 | EXISTS(n) 如果集合中存在第 n 个元素,则返回 TRUE;否则返回 FALSE。 |
| 2 | COUNT 返回集合当前包含的元素数量。 |
| 3 | LIMIT 检查集合的最大大小。 |
| 4 | FIRST 返回使用整数下标的集合中的第一个(最小)索引号。 |
| 5 | LAST 返回使用整数下标的集合中的最后一个(最大)索引号。 |
| 6 | PRIOR(n) 返回集合中在索引 n 之前的索引号。 |
| 7 | NEXT(n) 返回在索引 n 之后的索引号。 |
| 8 | EXTEND 向集合追加一个空元素。 |
| 9 | EXTEND(n) 向集合追加 n 个空元素。 |
| 10 | EXTEND(n,i) 向集合追加第 i 个元素的 n 个副本。 |
| 11 | TRIM 从集合末尾删除一个元素。 |
| 12 | TRIM(n) 从集合末尾删除 n 个元素。 |
| 13 | DELETE 删除集合中的所有元素,将 COUNT 设置为 0。 |
| 14 | DELETE(n) 从具有数字键的关联数组或嵌套表中删除第nth个元素。如果关联数组具有字符串键,则删除对应于键值的元素。如果n为空,则DELETE(n)不执行任何操作。 |
| 15 | DELETE(m,n) 从关联数组或嵌套表中删除范围m..n内的所有元素。如果m大于n或m或n为空,则DELETE(m,n)不执行任何操作。 |
集合异常
下表提供了集合异常及其引发的情况:
| 集合异常 | 引发的情况 |
|---|---|
| COLLECTION_IS_NULL | 尝试操作原子空集合。 |
| NO_DATA_FOUND | 下标指定已删除的元素,或关联数组中不存在的元素。 |
| SUBSCRIPT_BEYOND_COUNT | 下标超过集合中的元素数量。 |
| SUBSCRIPT_OUTSIDE_LIMIT | 下标超出允许的范围。 |
| VALUE_ERROR | 下标为空或不可转换为键类型。如果键定义为PLS_INTEGER范围,并且下标在此范围之外,则可能会发生此异常。 |