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).

  1. exos simples avec 1 boucle puis 2 boucles imbriquées (compter le nb d’occurrences, suppression de doublons, max tableau, etc)
  2. dichotomie
  3. récursion et pile d'appel (tours de hanoï ; min et max d'un tableau)
  4. plusieurs boucles a la suite
  5. 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 :

  1. factorielle.
  2. Fibonacci
  3. Triangle de pascal (optimisation pas tout garder)
  4. hanoi
  5. 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 :

cours ici https://www.cs.odu.edu/~zeil/cs361/latest/Public/recursionConversion/index.html#converting-recursive-algorithms-to-iteration

  1. factorielle (reprendre ce qu'on a vue et le formaliser)
  2. recursive simple : montrer que l'on peut sauver les paramètre et les remettre avec une todo list
  3. on peut faire mieux avec récursivité terminale et sa chaîne d'égalité.
  4. fonction 91 de Mc Carty. Récursif mais pas terminal. Montrer qu'on peut le rendre terminal.
  5. Fibonacci. par terminal. On utilise la pile. Puis on simplifie. A la fin fibo plus besoin de pile.
  6. Pas toujours le cas. Exemple : Ackermann.

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

  1. écrire comme récursion terminale (cf. wikipédia pour la fonction et https://fr.wikipedia.org/wiki/Récursion_terminale pour la définition)
  2. 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)$ :

  1. soit $m=0$ alors $A = n+1$
  2. soit $m>0$ et $n=0$ et alors $A = \text{Ack}(m-1, 1)$
  3. soit $m>0$ et $n>0$ et alors :
    1. $A = \text{Ack}(m, n-1)$
    2. $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 :

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 :

  1. on commence avec une TODO liste vide
  2. positionner les variables $m$ et $n$ à leurs valeurs et $A$ à 0
  3. on ajoute à la liste le triplet (1, m, n)
  4. tant que la TODO liste n'est pas vide :
    1. prendre le dernier item de la list (en le supprimant de la liste)
      1. si le premier élément de l'item vaut 1 alors on affecte :
        1. $m$ au second élément du triplet
        2. $n$ au troisième élément du triplet
      2. si le premier élément vaut 2 alors on affecte :
        1. $m$ au second élément du triplet
        2. $A$ à $n$
    2. On fait le calcul :
      1. si $m=0$ alors $A=n+1$
      2. si $m>0$ et $n=0$ alors on ajoute $(1, m-1, 1)$ à la TODO list
      3. si $m>0$ et $n>0$ alors :
        • on ajoute $(2, m-1)$ à la TODO list
        • on ajoute $(1, m, n-1)$ à la TODO list
  5. rendre $A$

TBD faire $A(2, 3)$

On peut même se passer de $A$ complètement :

  1. on commence avec une TODO liste vide
  2. positionner les variables $m$ et $n$ à leurs valeurs
  3. on ajoute à la liste l'entier $m$
  4. tant que la TODO liste n'est pas vide :
    1. prendre le dernier item de la list (en le supprimant de la liste) et en l'affectant à $m$
    2. 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$
  5. 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

  1. 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).