PL / SQL - Collections

Dans ce chapitre, nous aborderons les collections en PL / SQL. Une collection est un groupe ordonné d'éléments ayant le même type de données. Chaque élément est identifié par un indice unique qui représente sa position dans la collection.

PL / SQL fournit trois types de collection -

  • Tables indexées ou tableau associatif
  • Table imbriquée
  • Tableau de taille variable ou Varray

La documentation Oracle fournit les caractéristiques suivantes pour chaque type de collections -

Type de collecte Nombre d'éléments Type d'indice Dense ou clairsemé Où créé Peut être un attribut de type d'objet
Tableau associatif (ou table indexée) Sans bornes Chaîne ou entier Soit Uniquement dans le bloc PL / SQL Non
Table imbriquée Sans bornes Entier Commence dense, peut devenir clairsemé Soit en bloc PL / SQL, soit au niveau du schéma Oui
Tableau de taille de variable (Varray) Délimité Entier Toujours dense Soit en bloc PL / SQL, soit au niveau du schéma Oui

Nous avons déjà discuté de varray dans le chapitre 'PL/SQL arrays'. Dans ce chapitre, nous aborderons les tables PL / SQL.

Les deux types de tables PL / SQL, c'est-à-dire les tables indexées et les tables imbriquées ont la même structure et leurs lignes sont accessibles en utilisant la notation en indice. Cependant, ces deux types de tableaux diffèrent sous un aspect; les tables imbriquées peuvent être stockées dans une colonne de base de données et les tables indexées ne le peuvent pas.

Table d'indexation

Un index-by table (également appelée associative array) est un ensemble de key-valuepaires. Chaque clé est unique et sert à localiser la valeur correspondante. La clé peut être un entier ou une chaîne.

Une table indexée est créée à l'aide de la syntaxe suivante. Ici, nous créons unindex-by table nommée table_name, dont les clés seront de type subscript_type et les valeurs associées seront de type element_type

TYPE type_name IS TABLE OF element_type [NOT NULL] INDEX BY subscript_type; 
 
table_name type_name;

Exemple

L'exemple suivant montre comment créer une table pour stocker des valeurs entières avec des noms et plus tard, il imprime la même liste de noms.

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; 
/

Lorsque le code ci-dessus est exécuté à l'invite SQL, il produit le résultat suivant -

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.

Exemple

Les éléments d'une table indexée peuvent également être %ROWTYPE de n'importe quelle table de base de données ou %TYPEde n'importe quel champ de table de base de données. L'exemple suivant illustre le concept. Nous utiliserons leCUSTOMERS table stockée dans notre base de données comme -

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; 
/

Lorsque le code ci-dessus est exécuté à l'invite SQL, il produit le résultat suivant -

Customer(1): Ramesh  
Customer(2): Khilan  
Customer(3): kaushik     
Customer(4): Chaitali  
Customer(5): Hardik  
Customer(6): Komal  

PL/SQL procedure successfully completed

Tables imbriquées

UNE nested tableest comme un tableau unidimensionnel avec un nombre arbitraire d'éléments. Cependant, une table imbriquée diffère d'un tableau dans les aspects suivants -

  • Un tableau a un nombre déclaré d'éléments, mais pas une table imbriquée. La taille d'une table imbriquée peut augmenter de manière dynamique.

  • Un tableau est toujours dense, c'est-à-dire qu'il a toujours des indices consécutifs. Un tableau imbriqué est dense au départ, mais il peut devenir clairsemé lorsque des éléments en sont supprimés.

Une table imbriquée est créée à l'aide de la syntaxe suivante -

TYPE type_name IS TABLE OF element_type [NOT NULL]; 
 
table_name type_name;

Cette déclaration est similaire à la déclaration d'un index-by table, mais il n'y a pas INDEX BY clause.

Une table imbriquée peut être stockée dans une colonne de base de données. Il peut en outre être utilisé pour simplifier les opérations SQL où vous joignez une table à une seule colonne avec une table plus grande. Un tableau associatif ne peut pas être stocké dans la base de données.

Exemple

Les exemples suivants illustrent l'utilisation de la table imbriquée -

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; 
/

Lorsque le code ci-dessus est exécuté à l'invite SQL, il produit le résultat suivant -

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.

Exemple

Éléments d'un nested table peut aussi être un %ROWTYPEde n'importe quelle table de base de données ou% TYPE de n'importe quel champ de table de base de données. L'exemple suivant illustre le concept. Nous utiliserons la table CUSTOMERS stockée dans notre base de données comme -

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; 
/

Lorsque le code ci-dessus est exécuté à l'invite SQL, il produit le résultat suivant -

Customer(1): Ramesh  
Customer(2): Khilan  
Customer(3): kaushik     
Customer(4): Chaitali  
Customer(5): Hardik  
Customer(6): Komal  

PL/SQL procedure successfully completed.

Méthodes de collecte

PL / SQL fournit les méthodes de collecte intégrées qui facilitent l'utilisation des collections. Le tableau suivant répertorie les méthodes et leur objectif -

S. Non Nom et objectif de la méthode
1

EXISTS(n)

Renvoie TRUE si le nième élément d'une collection existe; sinon renvoie FALSE.

2

COUNT

Renvoie le nombre d'éléments qu'une collection contient actuellement.

3

LIMIT

Vérifie la taille maximale d'une collection.

4

FIRST

Renvoie les premiers (plus petits) numéros d'index d'une collection qui utilise les indices entiers.

5

LAST

Renvoie les derniers numéros d'index (les plus grands) d'une collection qui utilise les indices entiers.

6

PRIOR(n)

Renvoie le numéro d'index qui précède l'index n dans une collection.

sept

NEXT(n)

Renvoie le numéro d'index qui succède à l'index n.

8

EXTEND

Ajoute un élément nul à une collection.

9

EXTEND(n)

Ajoute n éléments nuls à une collection.

dix

EXTEND(n,i)

Ajoute ncopies du i ème élément dans une collection.

11

TRIM

Supprime un élément de la fin d'une collection.

12

TRIM(n)

Supprime n éléments de la fin d'une collection.

13

DELETE

Supprime tous les éléments d'une collection en définissant COUNT sur 0.

14

DELETE(n)

Supprime le nthélément d'un tableau associatif avec une clé numérique ou une table imbriquée. Si le tableau associatif a une clé de chaîne, l'élément correspondant à la valeur de clé est supprimé. Sin est nul, DELETE(n) ne fait rien.

15

DELETE(m,n)

Supprime tous les éléments de la plage m..nà partir d'un tableau associatif ou d'une table imbriquée. Sim est plus grand que n ou si m ou n est nul, DELETE(m,n) ne fait rien.

Exceptions de collection

Le tableau suivant fournit les exceptions de collection et quand elles sont déclenchées -

Exception de collection Élevé dans des situations
COLLECTION_IS_NULL Vous essayez d'opérer sur une collection atomiquement nulle.
AUCUNE DONNÉE DISPONIBLE Un indice désigne un élément qui a été supprimé, ou un élément inexistant d'un tableau associatif.
SUBSCRIPT_BEYOND_COUNT Un indice dépasse le nombre d'éléments dans une collection.
SUBSCRIPT_OUTSIDE_LIMIT Un indice est en dehors de la plage autorisée.
VALUE_ERROR Un indice est nul ou non convertible en type de clé. Cette exception peut se produire si la clé est définie comme unPLS_INTEGER plage, et l'indice est en dehors de cette plage.