CSC 3101 – Algorithmique et langage de programmation

Portail informatique

Calcul du plus grand commun diviseur (∼40mn – moyen – obligatoire)

Le but de cet exercice est d'écrire un programme permettant de calculer le plus grand commun diviseur (pgcd) de deux entiers.

Créez un projet nommé pgcd contenant une classe Pgcd, elle-même contenant une méthode main.

Affichez les arguments de votre programme après avoir spécifié que ses paramètres sont 243 576.

Modifiez votre programme de façon à vérifier que le nombre d'arguments est bien égal à 2. Si ce n'est pas le cas, votre programme doit afficher un message d'erreur et quitter à l'aide de la méthode System.exit(n), où n est le code de retour de votre programme. Si vous vous souvenez de votre cours d'introduction aux systèmes, vous devriez vous rappeler que toute valeur différente de 0 signifie que votre programme quitte avec une erreur. Vérifiez que votre programme se comporte correctement en modifiant les paramètres. Après vos tests, pensez à rétablir les arguments 243 et 576.

Les arguments sont donnés sous la forme de chaînes de caractères. Il faut donc les convertir en entier avant de pouvoir calculer leur plus grand commun diviseur. Pour cela, définissez deux variables entières nommées p et q, et utilisez Integer.parseInt(uneChaine) pour convertir la chaîne de caractères uneChaine en entier. Affichez les valeurs de p et q.

Un algorithme calculant le plus grand commun diviseur peut reposer sur la propriété suivante :

  • si p=0, alors pgcd(p ,q) = q,
  • si q=0, alors pgcd(p ,q) = p,
  • si p>q, alors pgcd(p ,q) = pgcd(p-q, q),
  • sinon : pgcd(p, q) = pgcd(p, q-p).

Mettez en œuvre un algorithme itératif (par opposition à récursif) utilisant cette propriété pour calculer le plus grand commun diviseur de p et q. Pour cela, utilisez une boucle while. Si vous obtenez pgcd(243, 576) = 9, vous avez gagné, bravo !

Estimez la complexité de cet algorithme de PGCD dans le pire des cas en fonction de n = p + q.
Pour répondre à cette question, nous devons estimer la complexité des étapes de l'algorithme. Ici, tout est en O ( 1 ) . Ensuite, il nous faut borner le nombre maximum de passages dans la boucle while en fonction de n. Étant donné que p et q décroissent strictement à chaque tour de boucle, le nombre maximum d'itérations est n. En particulier, ce nombre d'itérations est atteint pour p=n-1 et q=1.
La complexité finale est donc O ( n ) .

Il est possible d'implémenter un algorithme de PGCD plus efficace en remarquant que si p ≥ q, pgcd(p, q) = pgcd(q, p mod q). Implémenter cette variation du PGCD.

(Difficile) Estimez la complexité de ce nouvel algorithme dans le pire des cas en fonction de n = p + q. Pour cela, nous montrerons la propriété suivante : Soit ( p i , q i ) tel que p i q i et i le nombre d'itérations dans la boucle (si vous n'avez pas utilisé une structure récursive) avec ces deux entrées. Alors, nous avons q i F i b i avec Fib la suite de Fibonacci. On rappelle que F i b i i + φ i . Vous pouvez commencer par estimer la complexité de pgcd(Fibi+1,Fibi) en fonction de i, puis la traduire en une complexité en Fibi.
Pour calculer la complexité de pgcd(Fibi+1,Fibi), nous calculons Fibi+1Fibi+Fibi1Fibi1[Fibi]. Nous observons qu'à chaque étape, nous descendons d'un élément dans la suite de Fibonacci. On a donc une complexité linéaire en i. Comme Fibii+φi on obtient une complixité en O(log(Fibi)) quand i tend vers l'infini.
Ensuite, on peut prouver la propriété par induction.
  • Si i = 0, alors q0=0=Fib0b_1=0 \ge Fib_0
  • Si i = 1, alors q11=Fib1b_2 \ge 1 = Fib_1
  • Pour k ≥ 1 itérations, on a l'enchainement (pk+1,qk+1)(pk,qk)(pk1,qk1)(a_{k+1}, b_{k+1}) \rightarrow (a_{k}, b_{k}) \rightarrow (a_{k-1}, b_{k-1}) . Par conséquent, pk=qk+1,pk1=qk,qk1pk[qk]a_k = b_{k+1}, a_{k-1} = b_k, b_{k-1} \equiv a_k [b_k] . On a donc pk=q*qk+qk1,q1a_k = q*b_k+b_{k-1}, q \ge 1 et qk+1qk+qk1Fibk+Fibk1=Fibk+1b_{k+1} \ge b_k+b_{k-1} \ge Fib_{k-1} + Fib_{k-2} = Fib_{k}.
À partir de là, on a informellement pour i très grand : qiφib_i \ge \phi^i et donc ilog(qi)log(φ)i \le \frac{log(b_i)}{log(\phi)}. On en déduit que la complexité est en O(log(n))\mathcal{O}(log(n))
Notons que le pire des cas se produit quand les entrées sont deux nombres de Fibonacci consécutifs.