Une procédure est un ensemble d’instructions portant un nom. Pour définir une procédure, on utilise la syntaxe :
def nomprocedure(): instructions |
Une procédure est une nouvelle instruction, il suffit pour l’exécuter d’utiliser son nom. Par exemple, pour exécuter (on dit aussi appeler ou invoquer) une procédure appelée pr, il suffit d’écrire pr();. Les deux programmes suivants font la même chose. Le premier est écrit sans procédure :
Vous remarquez les instructions sous la portée de la procédure sont décalées d’une tabulation.
print ("Bonjour !") |
Et dans le deuxième, le
print
est placé dans la procédure
afficheBonjour
et cette procédure est appelée depuis le
script principal.
def afficheBonjour(): print ("Bonjour !") afficheBonjour() |
Vous pouvez définir autant de procédures que vous le voulez et vous pouvez appeler des procédures depuis des procédures :
def afficheBonjour(): print("Bonjour,") def afficheUn(): print("1") def afficheDeux(): print("2") def afficheUnEtDeux(): afficheUn() afficheDeux() def afficheAuRevoir(): print("Au revoir.") afficheBonjour() afficheUnEtDeux() afficheAuRevoir()
Ce programme affiche :
Bonjour, 1 2 Au revoir.
Regardez bien le programme suivant et essayez de déterminer ce qu’il affiche.
def procedure1(): print("debut procedure 1") print("fin procedure 1") def procedure2(): print("debut procedure 2") procedure1() print("fin procedure 2") def procedure3(): print("debut procedure 3") procedure1() procedure2() print("fin procedure 3") print("debut main") procedure2() procedure3() print("fin main")
La réponse est
debut main debut procedure 2 debut procedure 1 fin procedure 1 fin procedure 2 debut procedure 3 debut procedure 1 fin procedure 1 debut procedure 2 debut procedure 1 fin procedure 1 fin procedure 2 fin procedure 3 fin main
Il est possible de déclarer des variables à l’intérieur d’une procédure.
def nomprocedure(): i=0 print (i) |
Attention, ces variables ne sont accessibles que dans le corps de la procédure, cela signifie qu’elles naissent au moment de leur déclaration et qu’elles sont détruites une fois la dernière instruction de la procédure exécutée. C’est pour cela qu’on les apelle des variables locales. Une variable locale n’est visible qu’entre sa déclaration et la fin de la procédure. Par exemple, ce code est illégal :
def maProcedure(): a = b b = 'k' print(a) print(b) |
En effet, la variable b est déclarée dans la procédure
Main
, et n’est donc visible que dans cette même
procédure. Son utilisation dans
maProcedure
est donc
impossible. De même, la variable a est déclarée dans
maProcedure
, elle n’est pas visible dans le
Main
. Voici un exemple d’utilisation de variables locales :
def unADix(): for i in range(1, 11): print(i) unADix() |
Il est possible que la valeur d’une variable locale d’une procédure ne soit connue qu’au moment de l’appel de la procédure. Considérons le programme suivant :
i = int(input("Veuillez saisir un entier : ")) # Appel d'une procedure affichant la valeur de i. # ... |
Comment définir et invoquer une procédure
afficheInt
permettant d’afficher cet entier saisi par l’utilisateur ? Vous
conviendrez que la procédure suivante ne passera pas la compilation
def afficheInt(): print(i) |
En effet, la variable i est déclarée dans le
Main
, elle n’est donc
pas visible dans
afficheInt
. Pour y remédier, on définit
afficheInt
de la sorte :
def afficheInt(i): print(i) |
i est alors appelé un paramètre, il s’agit d’une variable
dont la valeur sera précisée lors de l’appel de la procédure. On peut
aussi considérer que i est une valeur inconnue, et qu’elle est
initialisée lors de l’invocation de la procédure. Pour
initialiser la valeur d’un paramètre, on place cette valeur entre les
parenthèses lors de l’appel de la procédure, par exemple :
afficheInt(4)
lance l’exécution de la procédure
afficheInt
en initialisant la valeur de i à 4. On dit
aussi que l’on passe en paramètre la valeur 4. La version
correcte de notre programme est :
def afficheInt(i): print(i) i = int(input("Veuillez saisir un entier : ")) afficheInt(i) |
Attention, notez bien que le i de
afficheInt
et le
i du
Main
sont deux variables différentes, la
seule chose qui les lie vient du fait que l’instruction
afficheInt(i)
initialise le i de
afficheInt
à la
valeur du i du
Main
. Il serait tout à fait possible
d’écrire :
def afficheInt(j): print(j) i = int(input("Veuillez saisir un entier : ")) afficheInt(i) |
Dans cette nouvelle version, l’instruction
afficheInt(i)
initialise le j de
afficheInt
à la valeur du i du
Main
.
Il est possible de passer plusieurs valeurs en paramètre. Par exemple,
la procédure suivante affiche la somme des deux valeurs passées en
paramètre :
def afficheSomme(a, b): print(a + b) |
L’invocation d’une telle procédure se fait en initialisant les
paramètres dans le même ordre et en séparant les valeurs par des
virgules, par exemple
afficheSomme(3, 5)
invoque
afficheSomme
en initialisant a à 3 et b à 5. Vous
devez initialiser tous les paramètres et vous devez placer les valeurs
dans l’ordre. Récapitulons :
def afficheSomme(a, b): print(a + b) i = int(input("Veuillez saisir deux entiers :\na = ")) j = int(input("b = ")) print("a + b = ") afficheSomme(i, j) |
La procédure ci-avant s’exécute de la façon suivante :
Veuillez saisir deux entiers : a = 3 b = 5 a + b = 8
Lors de l’appel
afficheInt(r)
de la procédure
public static void afficheInt(int i)
, r est le paramètre effectif et i
le paramètre formel. Notez bien que i et r sont deux
variables distinctes. Par exemple, qu’affiche le programme suivant ?
def incr(v): v += 1 i = 6 incr(i) print(i) |
La variable v est initialisée à la valeur de i, mais i et v sont deux variables différentes. Modifier l’une n’affecte pas l’autre.
Nous avons vu qu’un sous-programme appelant peut communiquer des valeurs au sous-programme appelé. Mais est-il possible pour un sous-programme appelé de communiquer une valeur au sous-programme appelant ? La réponse est oui. Une fonction est un sous-programme qui communique une valeur au sous-programme appelant. Cette valeur s’appelle valeur de retour, ou valeur retournée.
La syntaxe pour appeler une fonction est :
v = nomfonction(parametres) |
L’instruction ci-dessus place dans la variable v la valeur retournée
par la fonction
nomfonction
quand lui passe les paramètres
parametres
. Nous allons plus loin dans ce cours définir une
fonction
carre
qui retourne le carré de valeur qui lui est
passée en paramètre, alors l’instruction
v = carre(2) |
placera dans v le carré de 2, à savoir 4. On définira aussi une
fonction
somme
qui retourne la somme de ses paramètres, on
placera donc la valeur 2 + 3 dans v avec l’instruction
v = somme(2, 3) |
L’instruction servant à retourner une valeur est
return
. Cette instruction interrompt l’exécution de la
fonction et retourne la valeur placée immédiatement après. Par exemple,
la fonction suivante retourne toujours la valeur 1.
def un(): return 1 |
Lorsque l’on invoque cette fonction, par exemple
v = un() |
La valeur 1, qui est retournée par
un
est affectée à
v. On définit une fonction qui retourne le successeur de son
paramètre :
def successeur(i): return i + 1 |
Cette fonction, si on lui passe la valeur 5 en paramètre, retourne 6. Par exemple, l’instruction
v = successeur(5) |
affecte à v la valeur 6. Construisons Maintenant nos deux fonctions :
def carre(i): return i * i def somme(a, b): return a + b |
En Python, lorsque vous invoquez une fonction, toutes les valeurs des paramètres effectifs sont recopiés dans les paramètres formels. On dit dans ce cas que le passage de paramètre se fait par valeur. Vous ne pouvez donc, a priori, communiquer qu’une seule valeur au programme appelant. Par conséquent seule la valeur de retour vous permettra de communiquer une valeur au programme appelant.
Lorsque vous passez un tableau en paramètre, la valeur qui est recopiée dans le paramètre formel est l’adresse de ce tableau (l’adresse est une valeur scalaire). Par conséquent toute modification effectuée sur les éléments d’un tableau dont l’adresse est passée en paramètre par valeur sera repercutée sur le paramètre effectif (i.e. le tableau d’origine). Lorsque les modifications faites sur un paramètre formel dans un sous-programme sont repércutées sur le paramètre effectif, on a alors un passage de paramètre par référence.
from array import array def initTab(tableau): for i in range (0, len(tableau)): tableau[i] = i+1 def copieTab(original): copie = array('i', original) return copie tableau = array('i', range(0, 20)) initTab(tableau) autreTableau = copieTab(tableau) for x in autreTableau: print(x)
Nous retiendrons donc les règles d’or suivantes :