Cours Pls QL
Cours Pls QL
Cours Pls QL
Langage propre Oracle bas sur ADA Offre une extension procdurale SQL PL/SQL permet dutiliser un sous-ensemble du langage SQL des variables, des boucles, des alternatives, des gestions derreurs. PL/SQL permet galement de manipuler ligne par ligne les donnes retournes par une requte SQL.
Un programme PL/SQL est compos dun ou plusieurs blocs. Chaque bloc comporte 4 sections.
PL/SQL offre 2 grandes familles de type de donnes. Types scalaires (lignes ne retournant quune seule valeur) Types de base
INTEGER, NUMBER, FLOAT, REAL, , CHAR, VARCHAR2, DATE, BOOLEAN
Nom_variable nom_type
Variables bases %TYPE sert baser la variable sur une autre structure de donnes : une variable PL/SQL, un attribut dune relation, un curseur.
Variables bases Ex :
Base de donnes : Employ (nom VARCHAR2(30), Salaire NUMBER(8,2)) Bloc PL/SQL : DECLARE v_nom Employe.nom %TYPE v_new_nom v_nom %TYPE v_nom est en fait bas sur le type de lattribut nom de la table Employe, i.e. VARCHAR2(30) et v_new_nom est bas sur le type de v_nom.
Enregistrement
1.
2. Enregistrement bas sur une relation Utilisation de %ROWTYPE. Les enregistrements sont semblables, en terme de lignes, la structure dune relation. v_employe employe %ROWTYPE Accs au champ : nom_variable.nom_champi 3. Enregistrement bas sur un curseur Structure calque sur la liste du SELECT ramen par le curseur. nom_variable nom_curseur %ROWTYPE
Tableau
Une seule dimension index par BINARY INTEGER Elments de mme type
TYPE nom_type IS TABLE OF type_des_elements INDEX BY BINARY INTEGER nom_variable nom_type nom_variable(2)
Constantes
nom_constante CONSTANT type := valeur ;
Instructions SQL
INSERT, UPDATE, DELETE SELECT FROM WHERE Gestion des transactions : COMMIT, ROLLBACK
1.
:=
nom_variable := valeur ;
2. Valeur de rsultat dune requte
Alternative
IF condition THEN instruction1; ENDIF; IF condition THEN instruction1; ELSE instruction2 ; ENDIF; IF condition THEN instruction1; ELSIF condition2 then instruction2; ELSIF condition3... ELSE instructionN ; ENDIF;
Rptition
LOOP instruction; EXIT [WHEN condition] ; (ou IF condition THEN EXIT;) instruction; END LOOP; WHILE condition LOOP instruction; END LOOP ; FOR (variable_indice) IN [REVERSE] Min .. Max LOOP instruction; END LOOP;
Variable_indice : variable locale la boucle, elle na pas besoin dtre dclare REVERSE : va de Max Min Min, Max : constantes ou des variables dclares et affectes, le pas est obligatoirement de 1.
select expr1, expr2,... into var1, var2,... met des valeurs de la BD dans une ou plusieurs variables Le select ne doit renvoyer quune seule ligne Avec Oracle il nest pas possible dinclure un select sans into dans une procdure ;
Si le select renvoie plus dune ligne, une exception TOO_MANY_ROWS (ORA-01422) est leve Si le select ne renvoie aucune ligne, une exception NO_DATA_FOUND (ORA-01403) est leve
DECLARE v_nom emp.nome %TYPE; v_emp emp %ROWTYPE; BEGIN select nome into v_nom from employe where budget = 500; select * into v_emp from emp where budget = 500; END;
Un curseur permet de rcuprer et de traiter ligne par ligne un ensemble de lignes de la BD.
En PL/SQL, il existe 2 types de curseurs : les curseurs implicites : reprsente la zone mmoire utilise pour parser et excuter la requte
Ds linstant o lon excute un ordre SQL, il y a cration dun curseur. Un curseur est une zone de travail ( Context area ) qui contient toutes les infos relatives lordre SQL (texte source de la requte, forme traduite, plan dexcution, rsultat)
les curseurs explicites (grs par le programme PL/SQL, ils permettent de traiter une une les lignes ramenes avec un select),
Ouverture de curseur Alloue lespace mmoire suffisant et permet de positionner dventuels verrous OPEN nom_curseur ; OPEN nom_curseur(liste de paramtres rls) ; La requte est excute ce moment l. Fermeture du curseur CLOSE nom_curseur ; Traitement des lignes Distribution une une des lignes par lordre FETCH plac lintrieur dune boucle. FETCH nom_curseur INTO variable ;
%ISOPEN : renvoie VRAI si le curseur est ouvert %FOUND : renvoie VRAI si le dernier FETCH a renvoy 1 ligne %NOTFOUND %ROWCOUNT : renvoie le nombre de lignes traites par le curseur
Les curseurs implicites sont tous nomms SQL DECLARE nb_lignes integer; BEGIN delete from employe where dept = 10; nb_lignes := SQL%ROWCOUNT; ...
DECLARE CURSOR e IS SELECT * FROM Employe; v_nom Employ.nom %TYPE ; v_salaire Employe.salaire %TYPE ; BEGIN OPEN e; LOOP FETCH e INTO v_nom, v_salaire; EXIT WHEN e%NOTFOUND; Traitement END LOOP; CLOSE e; END;
Remarque : Quand on veut traiter une une, sans exception, les lignes retournes il faut utiliser une boucle FOR. FOR indice IN nom_curseur indice na pas besoin dtre dclar. OPEN et CLOSE sont excuts de manire implicite
DECLARE CURSOR e IS SELECT * FROM Employe; v_nom Employ.nom %TYPE v_salaire Employe.salaire %TYPE BEGIN FOR i IN e LOOP v_nom := i.nom; v_salaire := i.salaire; Traitement END LOOP; END;
On peut effectuer une modification (UPDATE ou DELETE) sur la ligne que renvoie le curseur grce la commande CURRENT OF . UPDATE | DELETE WHERE CURRENT OF nom_curseur La ligne courante dun curseur est dplace chaque appel de linstruction fetch On est parfois amen modifier la ligne courante pendant le parcours du curseur Pour cela on peut utiliser la clause where current of pour dsigner cette ligne courante dans un ordre LMD (insert, update, delete) Il est ncessaire davoir dclar le curseur avec la clause FOR UPDATE pour que le bloc compile FOR UPDATE [OF col1, col2,...]
Cette clause bloque toute la ligne ou seulement les colonnes spcifies Les autres transactions ne pourront modifier les valeurs tant que le curseur naura pas quitt cette ligne
Ex :
DECLARE CURSOR c IS SELECT * FROM Employe [FOR UPDATE OF Commission]; v_nom Employ.nom %TYPE v_salaire Employe.salaire %TYPE v_commission Employe.commission %TYPE BEGIN OPEN c; LOOP FETCH c INTO v_nom, v_salaire, v_commission; EXIT WHEN c %NOTFOUND; IF v_commission IS NULL THEN UPDATE Employe SET commission:=v_salaire * 0.1 WHERE CURRENT OF c ; END IF ; END LOOP; CLOSE e; COMMIT; END;
Une exception est une erreur qui survient durant une excution 2 types dexception : prdfinie par Oracle dfinie par le programmeur
DECLARE -- dfinitions de variables BEGIN -- Les instructions excuter EXCEPTION -- La rcupration des erreurs END;
DECLARE BEGIN EXCEPTION WHEN NO_DATA_Found THEN traitement ; WHEN Too_Many_Rows THEN traitement ; WHEN OTHERSTHEN traitement ; --optionnel END;
On la dclare et on effectue le traitement appropri, on dtecte lanomalie au cours du programme. DECLARE Nom_erreur EXCEPTION ; BEGIN IF anomalie THEN RAISE nom_erreur ; END IF ; ... EXCEPTION WHEN nom_erreur THEN traitement ; END ;
Lorsquune exception est dtecte, le traitement associ lexception est excut et ensuite on sort du bloc dans lequel lexception a t dfinie. Solution possible : plusieurs blocs imbriqus. BEGIN BEGIN IF exception THEN EXCEPTION END; BEGIN END; END;
On peut donner un code une exception : Nom_erreur EXCEPTION ; PRAGMA EXCEPTION_INIT(nom_erreur,code) ; Remarque : numro pris par Oracle : -20000 20995 Affichage de lerreur : ORA xxxxx : un message
CREATE TABLE err_test (nom VARCHAR2(10) PRIMARY KEY; numero NUMBER CONSTRAINT no_smallnb CHECK (numero>100)); INSERT INTO err_test VALUES (voiture,2); ORA 02290 : violation de contrainte(nom_uti.no_smallnb) de verification. Programme PL/SQL : BEGIN INSERT INTO EXCEPTION IF SQLCODE=-02290 AND SQLERRM=%no_smallnb THEN Affichage dun message adapt ELSE Affichage des autres erreurs
un dclencheur qui affiche le numro et le nom d'un employ que l'on veut supprimer de la table EMP CREATE OR REPLACE TRIGGER TRG_BDR_EMP BEFORE DELETE --avant supression ON EMP -- sur la table EMP FOR EACH ROW -- pour chaque ligne Declare LC$Chaine VARCHAR2(100); Begin dbms_output.put_line( 'Suppression de l''employ n' | | To_char( :OLD.empno ) || ' -> ' || :OLD.ename ) ; End ; Supprimons maintenant un employ delete from emp where empno = 7369 Suppression de l'employ n7369 -> SMITH
dsormais, tout nouvel employ devra avoir un numro suprieur ou gal 10000. Il faut donc interdire toute insertion qui ne reflte pas cette nouvelle directive
CREATE OR REPLACE TRIGGER TRG_BIR_EMP BEFORE INSERT ON EMP FOR EACH ROW Begin If :NEW.empno < 10000 Then RAISE APPLICATION_ERROR ( -20010, 'Numro employ infrieur 10000' ) ; End if ; End ; Tentons d'insrer un nouvel employ avec le numro 9999 insert into emp (empno, ename, job) values( 9999, 'Burger', 'CLERK' ) ; insert into emp (empno, ename, job) values( 9999, 'Burger', 'CLERK' ) ERREUR la ligne 1 : ORA-20010: Numro employ infrieur 10000
CREATE OR REPLACE TRIGGER TRG_BIUDR_EMP BEFORE INSERT OR UPDATE OR DELETE -- avant insertion, modification ou suppression ON EMP FOR EACH ROW Begin If INSERTING Then dbms_output.put_line( 'Insertion dans la table EMP' ) ; End if ; If UPDATING Then dbms_output.put_line( 'Mise jour de la table EMP' ) ; End if ; If DELETING Then dbms_output.put_line( 'Suppression dans la table EMP' ) ; End if ; End ;
Bloc anonyme ou nomm Un bloc anonyme PL/SQL est un bloc DECLARE BEGIN END comme dans les exemples prcdents
Le plus souvent, on cre plutt une procdure ou une fonction nomme pour rutiliser le code
create or replace PROCEDURE(<liste params>) IS -- dclaration des variables BEGIN -- code de la procdure END; Pas de DECLARE ; les variables sont dclares entre IS et BEGIN Si la procdure ne ncessite aucune dclaration, le code est prcd de IS BEGIN
create or replace FUNCTION(<liste params>) IS -- dclaration des variables BEGIN -- code de la fonction END;