- 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 表(即索引表和嵌套表)具有相同的结构,并且使用下标表示法来访问它们的行。然而,这两种类型的表在一方面有所不同;嵌套表可以存储在数据库列中,而索引表则不能。
按表索引
索引表(也称为关联数组)是一组键值对。每个键都是唯一的,用于定位对应的值。键可以是整数或字符串。
使用以下语法创建索引表。在这里,我们创建一个名为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或者m或n为空,则 DELETE(m,n)不执行任何操作。 |
集合例外
下表提供了集合异常以及引发异常的时间 -
集合异常 | 成长环境 |
---|---|
COLLECTION_IS_NULL | 您尝试对Atomics空集合进行操作。 |
没有找到数据 | 下标表示已删除的元素,或关联数组中不存在的元素。 |
SUBSCRIPT_BEYOND_COUNT 个 | 下标超过集合中元素的数量。 |
SUBSCRIPT_OUTSIDE_LIMIT | 下标超出了允许的范围。 |
VALUE_ERROR | 下标为空或不可转换为键类型。如果键定义为PLS_INTEGER范围,并且下标超出此范围,则可能会发生此异常。 |