In [1]:
Sys.command "ocaml -version";;
print_endline Sys.ocaml_version;;
Out[1]:
Out[1]:
In [7]:
let print f =
let r = Printf.printf f in
flush_all();
r
;;
Out[7]:
La question de programmation pour ce texte était donnée en page 6 :
Cet exercice consiste à observer le comportement des nombres en virgule flottante de la
machine.
Toutes les variables utilisées seront de type double
ou float
selon le langage choisi.
On écrit un programme qui évalue $$S = \sum_{i=0}^{\infty} \left(1 - \frac{1}{\rho}\right)^i$$
où $\rho$ est une puissance de $2$ suffisamment grande, par exemple $\rho = 2^{20}$.
Si on calcule naïvement $S_n = \sum_{i=0}^{n} (1 - \frac{1}{\rho})^i$, en ajoutant les termes un par un dans l'ordre des puissances croissantes, on remarque quand $n$ est suffisamment grand, $S_{n+1} = S_n \oplus (1 - \frac{1}{\rho})^{n+1}$
lorsqu’on calcule en double
(ou float
).
En évaluant $S$ sous la forme $$S = \sum_{j=0}^{\infty} \sum_{k=0}^{K-1} \left(1 - \frac{1}{\rho}\right)^{jK + k}$$ ce phénomène se produit pour des rangs plus grands que précedemment, l'approximation de $S$ calculée est donc meilleure. On pourra essayer plusieurs valeurs de $K$ ($1,10,100,1000,10000$) et comparer. $K = 1$ correspond au calcul naïf de $S$ ci-dessus. Optionnellement, le programme pourra également calculer une borne sur l’erreur com- mise sur $S$.
On est libre de choisir l'approche, mais ici il n'y a pas tellement de choix, on va utiliser des float
de OCaml.
Première méthode de calcul. On remarque qu'on est déjà malin, on n'utilise aucune exponentiation mais simplement un accumulateur terme
qui vaut $(1-1/\rho)^j$ pour les valeurs successives de $j$, et qu'on met à jour par une simple multiplication.
In [39]:
let premiercalcul (rho : float) (n : int) =
let somme = ref 0. in
let unmoinsunsurrho = 1. -. (1. /. rho) in
let terme = ref 1. in
for _ = 0 to n do (* terme = (1-1/rho)^j pour j = _ *)
somme := !somme +. !terme; (* somme = sum_k=0^j terme_k *)
terme := !terme *. unmoinsunsurrho; (* conserve l'invariant de boucle *)
done;
!somme
;;
Out[39]:
Voyons un exemple :
In [10]:
let rho = 2. ** 20.;;
Out[10]:
In [14]:
1. -. 1. /. rho;;
Out[14]:
Cette valeur est proche de $1$, mais strictement plus petite que $1$, et donc $(1-1/\rho)^i \to 0$ géométriquement vite, ainsi $S_n$ converge bien vers $S$ pour $n \to \infty$.
In [40]:
for n = 0 to 10 do
print "\nS_%i \t= %g" n (premiercalcul rho n);
done;
Out[40]:
In [42]:
for n = 1000 to 1010 do
print "\nS_%i \t= %g" n (premiercalcul rho n);
done;
Out[42]:
Deuxième méthode de calcul :
In [21]:
let deuxiemecalcul (rho : float) (k : int) (n : int) =
let somme = ref 0. in
let unmoinsunsurrho = 1. -. (1. /. rho) in
let terme = ref 1. in
for _ = 0 to n do
for _ = 0 to k-1 do
somme := !somme +. !terme;
terme := !terme *. unmoinsunsurrho;
done;
terme := !terme *. unmoinsunsurrho;
done;
!somme
;;
Out[21]:
Voyons le même exemple :
In [27]:
let valeurs_k = [|1; 10; 100; 1000; 10000|] in
for i = 0 to (Array.length valeurs_k) - 1 do
let k = valeurs_k.(i) in
print "\n\nFor K = %i ..." k;
for n = 0 to 10 do
print "\nS_%i \t= %g" n (deuxiemecalcul rho k n);
done;
done;
Out[27]:
Avec de grandes valeurs de $n$, on semble converger vers la limite $S$, d'autant plus vite que $K$ est grand.
In [38]:
let valeurs_n = [|100; 1000; 10000; 100000|] in
let valeurs_k = [|1; 10; 100; 1000; 10000|] in
for i = 0 to (Array.length valeurs_k) - 1 do
let k = valeurs_k.(i) in
print "\n\nFor K = %i ..." k;
for j = 0 to (Array.length valeurs_n) - 1 do
let n = valeurs_n.(j) in
print "\nS_%i \t= %g" n (deuxiemecalcul rho k n);
done;
done;
Out[38]:
Voilà pour la question obligatoire de programmation.
Bien-sûr, ce petit notebook ne se prétend pas être une solution optimale, ni exhaustive.
Vous auriez pu choisir de modéliser le problème avec une autre approche, mais je ne vois pas ce qu'on aurait pu faire différement.
C'est tout pour aujourd'hui les amis, allez voir ici pour d'autres corrections, et que la force soit avec vous !