Récursivité
Introduction Une fonction récursive est une fonction qui s'appelle elle-même. Directement (si la fonction P appelle directement P , on dit que la récursivité est directe). Indirectement à travers une ou plusieurs fonctions relais (si P appelle une fonction P1 , qui appelle une fonction P2 , ... , qui appelle une fonction Pn et qui enfin appelle P , on dit qu'il s'agit d'une récursivité indirecte).
Introduction Si f est une fonction comprenant un appel à elle même, soit directement ou indirectement, alors f est une fonction récursive. directement : indirectement : int f1( …) { . . . x= f1(…); } int f1( …) int f2(. . .) { . . . { . . . x= f2(…); x=f1(. . .); } }
Introduction La récursivité est une manière simple et élégante de résoudre certains problèmes algorithmiques. Elle permet: d'écrire des programmes beaucoup plus lisibles; d'écrire d'une manière très rapide (par rapport d’une manière itérative); d’utiliser le principe diviser-pour-résoudre.
Introduction La récursivité utilise toujours la pile du programme en cours. Dans une fonction récursive, toutes les variables locales sont stockées dans la pile, et empilées autant de fois qu'il y a d'appels récursifs. La pile se remplit progressivement, et si on ne fait pas attention on arrive à un "débordement de pile". Ensuite, les variables sont désempilées. Toute fonction récursive comporte une instruction (ou un bloc d'instructions) nommée "point terminal" ou "point d'appui" ou "point d'arrêt", qui indique que le reste des instructions ne doit plus être exécuté.
Définition récursive La fonction récursive est composée de deux partie: une partie strictement récursive et une partie non récursive (base) servant de point de départ à l'utilisation de la définiton récursive. Structure générale d’une fonction récursive : { if(/* !condition de convergence */) exit(1); if(/*condition d’arrêt*/) return(/*Ce qu’elle doit retourner*/); else appel récursif } Traitement { if(/* !condition de convergence */) exit(1); if(/*condition d’arrêt*/) return(/*Ce qu’elle doit retourner*/); else appel récursif } Traitement Traitement Traitement Traitement
Définition récursive On peut définir le factoriel d'un nombre n non négatif de deux manières: définition non récursive: définition récursive: n ! = n * n-1 * ... 2 * 1 n ! = n * (n-1) ! et 0 ! = 1
Définition récursive Expression récursive du problème : Condition d’arrêt : Convergence (vers la condition d’arrêt): Si n=1 ou n=0, alors on a convergé! Si n>1, alors la soustraction à l’étape suivante nous approche de n=1. Donc si n est une entier non négatif ça converge! n ! = n * n-1 * ... 2 * 1 = n*(n-1)! n = 1 ou n = 0
Fonctions récursives La grande question qui se pose dans une fonction récursive est celle de l'arrêt de la récursivité. La condition de sortie pour la fonction n! est 0!=1. Le choix de la condition - vous devrez être sûr qu'elle soit validée à un moment ou à un autre sinon c'est comme si vous créez une boucle infinie sans condition de sortie ! La syntaxe la plus générale d'une fonction récursive est : <type_de_retour> <nomFct>(<args>){ [déclaration de variables] [test d'arrêt] [suite d'instructions] [appel de <nomFct>(<args'>)] return <résultat>; }
Fonctions récursives Premier exemple - calcul de n!. (0!=1!=1 et n! = n × (n-1)!). De manière itérative, on écrit : unsigned long factorielle(int n){ unsigned long f = 1; for(int k = n; k > 1; k--) f *= k; return f; } le calcul par accumulation du produit dans la variable f
Fonctions récursives Premier exemple - calcul de n!. (0!=1!=1 et n! = n × (n-1)!). De manière récursive, on peut écrire : Hypothèse de convergence:n>=0 unsigned long fact(int n){ if (n < 0) exit (EXIT_FAILURE); else if(n == 1 || n == 0) return 1L; else return n * fact(n-1); } Cas de base (Condition d’arrêt) La relation de récurrence (appel récursif)
Fonctions récursives Une fonction de ce type possède deux parcours: Empilement des appels récursifs (fact(3)) ou la phase de descente
Fonctions récursives Dépilement des appels récursifs (fact(3)) ou la phase de remontée
Fonctions récursives La phase de descente et de remontée dans la pile des appels de la fonction récursive: Au moment de la remontée, où la condition de sortie est vraie les appels enregistrés sont dépilés.
Fonctions récursives La récursivité ne marche que si on ne fait pas déborder la pile d'appels. Le problème fondamental de l'informatique de pouvoir prouver qu'une fonction (ou un algorithme) termine. unsigned long fact(int n){ if (n < 0) exit (EXIT_FAILURE); else if(n == 1 || n == 0) return 1L; else return n * fact(n+1); } La fonction ne termine pas
Variables locales, arguments de fonctions Fonctions récursives Variables locales, arguments de fonctions Lorsqu'une fonction récursive définit des variables locales, un exemplaire de chacune d'entre elles est crée à chaque appel récursif de la fonction. Il en est de même des arguments des fonctions.
Variables locales, arguments de fonctions Fonctions récursives Variables locales, arguments de fonctions Exemple - considérons une fonction void miroir() qui lit caractère par caractère une chaîne terminée par '?' et l'affiche dans l'ordre inverse de celui de la lecture. Version non récursive #include <stdio.h> void miroir() { char s[20],c; int i=0; while( (c=getchar())!='?') s[i++]=c; s[i]='\0'; while (--i>=0) putchar (s[i]); } void main() { printf("Entrer les caracteres avec ? a la fin.\n"); miroir(); Entrer les caracteres avec ? a la fin. abcd? dcba
Variables locales, arguments de fonctions Fonctions récursives Variables locales, arguments de fonctions Exemple - considérons une fonction void miroir() récursive qui lit caractère par caractère une chaîne terminée par '?' et l'affiche dans l'ordre inverse de celui de la lecture. Version récursive sans vriable locale -> un tableau pour stocker tous les caractères jusqu'au caractère '?' pour ensuite les afficher dans l'ordre inverse. void miroir() { int c = getchar(); if (c != '?') { miroir(); printf("%c", c); } void main() { printf("Entrer les caracteres avec ? a la fin.\n"); miroir(); Entrer les caracteres avec ? a la fin. abcd? dcba
Variables locales, arguments de fonctions Fonctions récursives Variables locales, arguments de fonctions miroir() c=‘1’ miroir() c=‘2’ miroir() c=‘3’ miroir() c=‘?’ destruction de c retour affichage de ‘3’ affichage de ‘2’ affichage de ‘1’ Exemple d'exécution
Dangers et précautions Fonctions récursives Dangers et précautions Dépassement de capacité Il est d'usage de choisir un type approprié, même si vous êtes certains que le type que vous avez choisi ne sera jamais dépassé. Utilisez tant que possible une variable pouvant contenir de plus grandes données. Ceci s'applique à tous types de données. Evitez le type int si vous travaillez avec une fonction récursive.
Dangers et précautions Fonctions récursives Dangers et précautions Débordement de pile (Stack Overflow) Dans la pile sont non seulement stockés les valeurs des variables de retour mais aussi les adresses des fonctions entre autres choses, les données sont nombreuses et un débordement de la pile peut très vite arriver ce qui provoque sans conteste une sortie anormale du programme. Dans l'exemple de la fonction factoriel, il nous faut (en arrondissant) environ 135000 appels récursifs pour faire exploser la pile. Si vous êtes presque sûr de dépasser ce genre de limites, préférez alors une approche itérative plutôt qu'une approche récursive du problème.
Dangers et précautions Fonctions récursives Dangers et précautions Un piège subtil : les nombres de Fibonacci Exemple - écrire une fonction qui calcule le n-ième terme de la suite de Fibonacci, définie par F0 = 0, F1 = 1 et pour n≥ 2, Fn = Fn-1 + Fn-2. Le programme marche, il termine. Le problème se situe dans le nombre d'appels à la fonction. On fait un nombre exponentiel d'appels à la fonction. long fib(int n){ if(n <= 1) return n; // cas de base else return fib(n-1)+fib(n-2); }
Dangers et précautions Fonctions récursives Dangers et précautions Un piège subtil : les nombres de Fibonacci L'arbre des appels pour cette fonction, qui généralise la pile des appels :
Dangers et précautions Fonctions récursives Dangers et précautions Un piège subtil : les nombres de Fibonacci Une façon de calculer Fn qui ne coûte que n appels est la suivante. On calcule les valeurs du couple (Fi, Fi+1). long fib(int n){ int i, u, v, w; // u = F(0); v = F(1) u = 0; v = 1; for(i = 2; i <= n; i++){ // u = F(i-2); v = F(i-1) w = u+v; u = v; v = w; } return v;
Fonctions récursives Diviser pour résoudre Quand on ne sait pas résoudre un problème, on essaie de le couper en morceaux qui seraient plus faciles à traiter. Exemple - Recherche d'une racine par dichotomie On suppose que f :[a, b] → R est continue et telle que f(a) < 0, f(b) > 0. Il existe une racine x0 de f dans l'intervalle [a, b], qu'on veut déterminer de sorte que |f(x0)| ≤ ε pour ε donné. On calcule f((a+b)/2). En fonction de son signe, on explore [a, m] ou [m, b].
Fonctions récursives Diviser pour résoudre Exemple #include <stdio.h> 1/2 #include <math.h> double f(double x){ return x*x*x-2; } double racineDicho(double a, double b, double eps){ double m = (a+b)/2; double fm = f(m); if(abs(fm) <= eps) return m; if(fm < 0) // la racine est dans [m, b] return racineDicho(m, b, eps); else // la racine est dans [a, m] return racineDicho(a, m, eps); Programmer la fonction f. La fonction qui cherche la racine
Fonctions récursives Diviser pour résoudre Exemple void main() 2/2 { double a,b,eps; printf("a="); scanf("%lf",&a); printf("b="); scanf("%lf",&b); printf("eps="); scanf("%lf",&eps); printf("La racine=%lf\n",racineDicho(a,b,eps)); } a=1 b=6 eps=0.01 La racine=1.312500
Fonctions récursives terminales Lorsqu'il s'agit de faire de plus profondes récursions, un autre type de fonction récursive existe - c'est la récursivité terminale. Une grille de Loto contient 49 numéros dont seulement 6 peuvent êtres tirés. Le calcule le factoriel de 49 donne: 49! = 49x48x47 ... 8x7x6! Le résultat obtenu est d'une grandeur inimaginable !
Fonctions récursives terminales Avec une récursivité normale, on a (49-6)x2 passages soit 43 appels empilés l'un après l'autre sans compter qu'il faut également remonter tous les appels ce qui nous fait un total de 86 passages. La perte de temps sur des récursions encore plus profondes et qui risqueraient par ailleurs de faire exploser la pile ce qui conduirait irrémédiablement au plantage du programme ! La récursion terminale - c'est une récursion avec uniquement une phase de descente, sans remontée. De cette manière on économise l'utilisation de la pile du programme, et on gagne du temps en exécution.
Fonctions récursives terminales Ceci est possible car la dernière expression return factoriel_terminale (...) renvoie directement la valeur obtenue par l'appel récursif courant, sans qu'il n'y ait d'autres opérations à faire, ce qui n'est pas le cas dans la fonction récursive simple, où l'on multiplie n par le retour de la fonction. Les appels de la fonction n'ont pas besoin d'êtres empilés car l'appel suivant remplace simplement l'appel précédent dans le contexte d'exécution.
Fonctions récursives terminales Fonction récursive terminale: La dernière instruction est la fin de l'appel courant de la fonction et donc l'appel suivant peut prendre la place de la précédente car le résultat se trouve dans le second argument. unsigned long factoriel_terminal (int n, unsigned long result){ if (n < 0) { exit (EXIT_FAILURE); } if (n == 1) { return result; } else if (n == 0) { return 1L; } return factoriel_terminal (n - 1, n * result); } L’argument supplémentaire, est un passage obligatoire pour créer une récursivité terminale.
Fonctions récursives terminales La représentation de parcours de la fonction
Quand utiliser la récursivité Fonctions récursives Quand utiliser la récursivité Quand il existe une définition récursive claire. Quand la récursivité est plus simple que la version itérative. Quand on a besoin d’un gain en performance possible grâce à une formulation récursive habile. Généralement rapide et simple pour le développement. Temps de mise au point («débogage ») peut être long. Elle est lourde à l’exécution