Projet : Écrire et prouver des algorithmes Itératif et récursif
TBD : à faire propre
TBD premières séries d'exercice permettant comprendre comment créer des algorithmes simple de façon itérative, r´´cursives et comment passer de l'un à l'autre.
Écrire des algorithmes simples en pseudo-code (ou en python) pour résoudre des problèmes algorithmiques.
TBD : écrire des algos pour résoudre des problèmes. TBD essence même d'un algorithme c'est la boucle et les if/them/else sinon fait tout le temps la même chose (addition, etc). Le problème est de transformer une idée en code et en succession logique d'arguments (conditions) et d'action (instruction) tant que ce n'est pas résolu (boucles).
- exos simples avec 1 boucle puis 2 boucles imbriquées (compter le nb d’occurrences, suppression de doublons, max tableau, etc)
- dichotomie
- récursion et pile d'appel (tours de hanoï ; min et max d'un tableau)
- plusieurs boucles a la suite
- récursion et boucles
voir les parties complexité et remettre si possible les réponse dans cette partie.
exemples :
Maximum d'un tableau
TBD mettre en récursif en exercice.
On va voir 2 algorithmes pour calculer la valeur maximum d'un tableau de réels.
Algorithme récursif
def maximum(tab):
if len(tab) == 1:
return tab[0]
x = maximum(tab[1:])
if tab[0] > x
return tab[0]
else:
return x
On a utilisé la copie de sous-listes (slicing) de python : tab[1:]
est la liste contenant les éléments de tab
à partir de l'index 1 (donc sans le premier élément de tab
).
Finitude
À chaque récursion, la taille du tableau diminue strictement. La récursion s'arrêtera donc forcément lorsque le tableau sera de taille 1.
Preuve
Par récurrence sur la longueur d'un tableau. On vérifie que l'algorithme fonctionne pour une longueur de tableau valant 1, puis on effectue preuve par récurrence sur la longueur du tableau.
euclide
Faire euclide récursif.
Algorithmes itératif
Algorithmes récursif
terminale/ pas terminale. https://web4.ensiie.fr/~dubois/recursivite.pdf TBD :
- factorielle.
- Fibonacci
- Triangle de pascal (optimisation pas tout garder)
- hanoi
- lancers de nd6 avec nombre max de récursion à trouver.
Rendre itératif des algorithmes récursifs
TBD grace à une todo list (aka une pile) https://www.youtube.com/watch?v=HXNhEYqFo0o
TBD :
- factorielle (reprendre ce qu'on a vue et le formaliser)
- recursive simple : montrer que l'on peut sauver les paramètre et les remettre avec une todo list
- on peut faire mieux avec récursivité terminale et sa chaîne d'égalité.
- fonction 91 de Mc Carty. Récursif mais pas terminal. Montrer qu'on peut le rendre terminal.
- Fibonacci. par terminal. On utilise la pile. Puis on simplifie. A la fin fibo plus besoin de pile.
- Pas toujours le cas. Exemple : Ackermann.
- recursivité terminale = qu'une suite d'égalité. C'est donc super.
- https://www.lirmm.fr/~dony/notesCours/c4.s.pdf
- https://www.enseignement.polytechnique.fr/informatique/INF321/Amphis12/amphi5.pdf
- https://web4.ensiie.fr/~dubois/recursivite.pdf fact est récursive primitive.L'écrire de façon itérative.
Fonction 91 de McCarty
Dans le même ordre d'idée que la fonction de Takeuchi.
TBD : https://fr.wikipedia.org/wiki/Fonction_91_de_McCarthy
TBD a écrire algo avec récursion puis : essayer de se passer de f(f(x))
- écrire comme récursion terminale (cf. wikipédia pour la fonction et https://fr.wikipedia.org/wiki/Récursion_terminale pour la définition)
- avec un tant que utiliser https://www.cs.odu.edu/~zeil/cs361/latest/Public/recursionConversion/index.html#converting-recursive-algorithms-to-iteration pour faire https://www.corsi.univr.it/documenti/OccorrenzaIns/matdid/matdid779820.pdf
récursivité terminale = qu'une suite d'égalité. C'est donc super.
TBD et quelle est sa valeur ?
Ackermann, le retour
Essayons de voir comment écrire l'algorithme d'Ackermann sans toutes ces récurrences, comme on l'a fait avec la fonction 91.
Pour calculer $\text{Ack}(m, n)$ :
- soit $m=0$ alors $A = n+1$
- soit $m>0$ et $n=0$ et alors $A = \text{Ack}(m-1, 1)$
- soit $m>0$ et $n>0$ et alors :
- $A = \text{Ack}(m, n-1)$
- $A = \text{Ack}(m-1, A)$
On voit que ce n'est pas une formulation récursive terminale à cause du troisième cas. En 3.2 :
- on utilise la valeur de $A$ calculée en 3.1
- on utilise la valeur que valait $m$ avant 3.1
Il faut donc se rappeler de $m$ pour le calcul de 3.2 tout en utilisant la valeur de $A$ calculée précédemment. Pour ce genre de récursion, on peut utiliser une TODO list qui nous permet de nous rappeler toutes les tâches à effectuer et des variables à sauvegarder :
- on commence avec une TODO liste vide
- positionner les variables $m$ et $n$ à leurs valeurs et $A$ à 0
- on ajoute à la liste le triplet (1, m, n)
- tant que la TODO liste n'est pas vide :
- prendre le dernier item de la list (en le supprimant de la liste)
- si le premier élément de l'item vaut 1 alors on affecte :
- $m$ au second élément du triplet
- $n$ au troisième élément du triplet
- si le premier élément vaut 2 alors on affecte :
- $m$ au second élément du triplet
- $A$ à $n$
- si le premier élément de l'item vaut 1 alors on affecte :
- On fait le calcul :
- si $m=0$ alors $A=n+1$
- si $m>0$ et $n=0$ alors on ajoute $(1, m-1, 1)$ à la TODO list
- si $m>0$ et $n>0$ alors :
- on ajoute $(2, m-1)$ à la TODO list
- on ajoute $(1, m, n-1)$ à la TODO list
- prendre le dernier item de la list (en le supprimant de la liste)
- rendre $A$
TBD faire $A(2, 3)$
On peut même se passer de $A$ complètement :
- on commence avec une TODO liste vide
- positionner les variables $m$ et $n$ à leurs valeurs
- on ajoute à la liste l'entier $m$
- tant que la TODO liste n'est pas vide :
- prendre le dernier item de la list (en le supprimant de la liste) et en l'affectant à $m$
- On a un choix selon les valeurs de $m$ et $n$ :
- si $m=0$ alors $n=n+1$
- si $m>0$ et $n=0$ alors (récursion terminale):
- $n = 1$
- ajoute l'entier $m-1$ à la TODO List
- $m>0$ et $n>0$ alors :
- ajoute l'entier $m-1$ à la TODO List
- ajoute l'entier $m$ à la TODO List
- $n=n-1$
- rendre $A$
Implémentation en python en utilisant une liste comme TODO-list :
def Ack(m, n):
s = [m]
while (s):
m = s.pop()
if m == 0:
n += 1
elif n == 0:
s.append(m - 1)
n = 1
else:
s.append(m - 1)
s.append(m)
n -= 1
return n
pas récursivité terminale = il faut faire des trucs en plus de la récursion. Il faut se rappeler de que l'on veut faire. Avec une TODO list (faire exemple avec ack petit) = pile en informatique faire un item de la todo list = dépile.
truc à faire = empile https://www.cs.odu.edu/~zeil/cs361/latest/Public/recursionConversion/index.html#conversion-using-stacks
- curryfication puis decurryfication A(m, n) = A(0, n') A'(s, n) = A'(s[:-1], A(s[-1], n)) A'([m], n) = A(m, n) par récurrence sur m+n = k
TBD faire la preuve que c'est ok (voir https://stackoverflow.com/a/54356919)
Notez que tout algorithme récursif peut s'écrire de façon itérative avec une TODO-list (une pile).