PL / SQL - Curseurs

Dans ce chapitre, nous aborderons les curseurs en PL / SQL. Oracle crée une zone de mémoire, appelée zone de contexte, pour le traitement d'une instruction SQL, qui contient toutes les informations nécessaires au traitement de l'instruction; par exemple, le nombre de lignes traitées, etc.

UNE cursorest un pointeur vers cette zone de contexte. PL / SQL contrôle la zone de contexte via un curseur. Un curseur contient les lignes (une ou plusieurs) renvoyées par une instruction SQL. L'ensemble de lignes que contient le curseur est appeléactive set.

Vous pouvez nommer un curseur afin qu'il puisse être référencé dans un programme pour récupérer et traiter les lignes renvoyées par l'instruction SQL, une à la fois. Il existe deux types de curseurs -

  • Curseurs implicites
  • Curseurs explicites

Curseurs implicites

Les curseurs implicites sont automatiquement créés par Oracle chaque fois qu'une instruction SQL est exécutée, lorsqu'il n'y a pas de curseur explicite pour l'instruction. Les programmeurs ne peuvent pas contrôler les curseurs implicites et les informations qu'ils contiennent.

Chaque fois qu'une instruction DML (INSERT, UPDATE et DELETE) est émise, un curseur implicite est associé à cette instruction. Pour les opérations INSERT, le curseur contient les données qui doivent être insérées. Pour les opérations UPDATE et DELETE, le curseur identifie les lignes qui seraient affectées.

En PL / SQL, vous pouvez faire référence au curseur implicite le plus récent comme SQL cursor, qui a toujours des attributs tels que %FOUND, %ISOPEN, %NOTFOUND, et %ROWCOUNT. Le curseur SQL a des attributs supplémentaires,%BULK_ROWCOUNT et %BULK_EXCEPTIONS, conçu pour être utilisé avec le FORALLdéclaration. Le tableau suivant fournit la description des attributs les plus utilisés -

S. Non Attribut et description
1

%FOUND

Renvoie TRUE si une instruction INSERT, UPDATE ou DELETE affectait une ou plusieurs lignes ou si une instruction SELECT INTO renvoyait une ou plusieurs lignes. Sinon, il renvoie FALSE.

2

%NOTFOUND

Le contraire logique de% FOUND. Elle renvoie TRUE si une instruction INSERT, UPDATE ou DELETE n'affecte aucune ligne ou si une instruction SELECT INTO ne renvoie aucune ligne. Sinon, il renvoie FALSE.

3

%ISOPEN

Renvoie toujours FALSE pour les curseurs implicites, car Oracle ferme automatiquement le curseur SQL après avoir exécuté son instruction SQL associée.

4

%ROWCOUNT

Renvoie le nombre de lignes affectées par une instruction INSERT, UPDATE ou DELETE, ou renvoyées par une instruction SELECT INTO.

Tout attribut de curseur SQL sera accessible comme sql%attribute_name comme indiqué ci-dessous dans l'exemple.

Exemple

Nous utiliserons la table CUSTOMERS que nous avons créée et utilisée dans les chapitres précédents.

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 | 
+----+----------+-----+-----------+----------+

Le programme suivant mettra à jour le tableau et augmentera le salaire de chaque client de 500 et utilisera le SQL%ROWCOUNT attribut pour déterminer le nombre de lignes affectées -

DECLARE  
   total_rows number(2); 
BEGIN 
   UPDATE customers 
   SET salary = salary + 500; 
   IF sql%notfound THEN 
      dbms_output.put_line('no customers selected'); 
   ELSIF sql%found THEN 
      total_rows := sql%rowcount;
      dbms_output.put_line( total_rows || ' customers selected '); 
   END IF;  
END; 
/

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

6 customers selected  

PL/SQL procedure successfully completed.

Si vous vérifiez les enregistrements dans la table des clients, vous constaterez que les lignes ont été mises à jour -

Select * from customers;  

+----+----------+-----+-----------+----------+ 
| ID | NAME     | AGE | ADDRESS   | SALARY   | 
+----+----------+-----+-----------+----------+ 
|  1 | Ramesh   |  32 | Ahmedabad |  2500.00 | 
|  2 | Khilan   |  25 | Delhi     |  2000.00 | 
|  3 | kaushik  |  23 | Kota      |  2500.00 | 
|  4 | Chaitali |  25 | Mumbai    |  7000.00 | 
|  5 | Hardik   |  27 | Bhopal    |  9000.00 | 
|  6 | Komal    |  22 | MP        |  5000.00 | 
+----+----------+-----+-----------+----------+

Curseurs explicites

Les curseurs explicites sont des curseurs définis par le programmeur pour obtenir plus de contrôle sur le context area. Un curseur explicite doit être défini dans la section déclaration du bloc PL / SQL. Il est créé sur une instruction SELECT qui renvoie plusieurs lignes.

La syntaxe pour créer un curseur explicite est -

CURSOR cursor_name IS select_statement;

Travailler avec un curseur explicite comprend les étapes suivantes -

  • Déclaration du curseur d'initialisation de la mémoire
  • Ouverture du curseur pour allouer la mémoire
  • Récupération du curseur pour récupérer les données
  • Fermer le curseur pour libérer la mémoire allouée

Déclaration du curseur

La déclaration du curseur définit le curseur avec un nom et l'instruction SELECT associée. Par exemple -

CURSOR c_customers IS 
   SELECT id, name, address FROM customers;

Ouverture du curseur

L'ouverture du curseur alloue la mémoire pour le curseur et le rend prêt à récupérer les lignes retournées par l'instruction SQL. Par exemple, nous allons ouvrir le curseur défini ci-dessus comme suit -

OPEN c_customers;

Récupération du curseur

La récupération du curseur implique d'accéder à une ligne à la fois. Par exemple, nous allons récupérer les lignes du curseur ouvert ci-dessus comme suit -

FETCH c_customers INTO c_id, c_name, c_addr;

Fermer le curseur

Fermer le curseur signifie libérer la mémoire allouée. Par exemple, nous fermerons le curseur ouvert ci-dessus comme suit -

CLOSE c_customers;

Exemple

Voici un exemple complet pour illustrer les concepts de curseurs explicites & minua;

DECLARE 
   c_id customers.id%type; 
   c_name customer.name%type; 
   c_addr customers.address%type; 
   CURSOR c_customers is 
      SELECT id, name, address FROM customers; 
BEGIN 
   OPEN c_customers; 
   LOOP 
   FETCH c_customers into c_id, c_name, c_addr; 
      EXIT WHEN c_customers%notfound; 
      dbms_output.put_line(c_id || ' ' || c_name || ' ' || c_addr); 
   END LOOP; 
   CLOSE c_customers; 
END; 
/

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

1 Ramesh Ahmedabad  
2 Khilan Delhi  
3 kaushik Kota     
4 Chaitali Mumbai  
5 Hardik Bhopal   
6 Komal MP  
  
PL/SQL procedure successfully completed.