Aller au contenu

2.8 Codage des non-entiers⚓︎

image

1. Notation binaire des décimaux⚓︎

a. Écriture de position⚓︎

Comme la notation décimale, la notation binaire permet aussi de représenter les nombres à virgule.

En notation décimale, les chiffres de gauche représentent les unités, les dizaines et ainsi de suite et ceux à droite de la virgule, les dixièmes, les centièmes etc.

decomposition decimal

De même, en notation binaire, les chiffres de droite représentent des demis, des quarts, des huitièmes etc.

decomposition binaire

Exercice

Trouver les nombres dont la représentation en binaire est :
▶ 1 001,101 1
▶ 10 101,011 101

▶ 1 001,101 1 code 9,6875
▶ 10 101,011 101 code 21,453125

b. De l’écriture décimale à la notation binaire⚓︎

Écriture d'un nombre décimal en binaire ❤

conversion de 12, 6875 en binaire
1⃣ Conversion de 12 donne \((1100)_2\)
2⃣ On effectue successivement des multiplications par 2 de la partie décimale, on conserve les parties entières

image

▶ Donc la conversion de 0,6875 en binaire est \((0,1011)_2\)

Exercice 1

Donner l'écriture binaire de 7,09375.

  • partie entière : \(7 = 111_2\)
  • partie décimale :
    • \(0,09375 \times 2 = \textbf{0},1875\)
    • \(0,1875 \times 2 = \textbf{0},375\)
    • \(0,375 \times 2 = \textbf{0},75\)
    • \(0,75 \times 2 = \textbf{1},5\)
    • \(0,5 \times 2 = \textbf{1}\)

Donc \(7,09375=111,00011_2\)

Exercice 2

Donner l'écriture binaire de 0,2.

  • partie entière : \(0 = 0_2\)
  • partie décimale :
    • \(0,2 \times 2 = \textbf{0},4\)
    • \(0,4 \times 2 = \textbf{0},8\)
    • \(0,8 \times 2 = \textbf{1},6\)
    • \(0,6 \times 2 = \textbf{1},2\)
    • \(0,2 \times 2 = \textbf{0},4\)
    • et cela continue...

Le nombre 0,2 n'admet pas d'écriture binaire finie.

Problèmes :

  • le processus de "conversion" ne s'arrête pas, nous obtenons une écriture binaire infinie périodique.
Remarque

certains nombres (par exemple : \(7/11 = 0.63636363 …\)) ont une écriture décimale infinie périodique. Ils auront alors une écriture binaire infinie périodique.
Et certains nombres ont une écriture binaire infinie périodique, alors même que leur écriture décimale est finie (exemple : 0.2)

Moralité :

Ce système d'écriture ne marche pas bien.

Conclusion :

Certains nombres n'admettent pas une écriture binaire finie. Or la mémoire d'un ordinateur, quelqu'il soit, est toujours finie. Certains nombres ne peuvent donc pas être représentés correctement en machine : c'est une impossibilité théorique. Cela amène à des comportements étranges :

🐍 Script Python
>>> 0.1 + 0.2
0.30000000000000004

Remarque : parmi les nombres décimaux à un chiffre après la virgule (0,1 0,2 0,3 ...) seul 0,5 admet une écriture binaire finie ! Tous les autres ont une représentation en machine qui n'en donne qu'une valeur approchée.

2. Norme IEE754 (Hors programme)⚓︎

a. Notation scientifique⚓︎

En notation décimale, elle consiste à exprimer le nombre sous la forme \(\pm a \times 10^n\)\(\pm\) est appelé signe, \(a\) est un nombre décimal de l’intervalle \([1 , 10[\) appelé mantisse (ou significande) et \(n\) est un entier relatif appelé exposant.

❓ Quel est la notation scientifique de \(105,745\)

Réponse

\(1,05745 \times 10^2\)

❓ Quel est la notation scientifique de \(0, 0745\)

Réponse

\(7,45 \times 10^{-2}\)

De même, en notation binaire, tout nombre s’exprime sous la forme \(\pm a \times 2^n\)\(\pm\)\(\pm\) est le signe, \(a\) est un nombre de l’intervalle \([(1)_2 , (10)_2[\) appelé mantisse et \(n\) est un entier relatif appelé exposant.

❓ Quel est la notation scientifique de \(1011, 0111 101\)

Réponse

\(1,0110111 101 \times 2^3\)

❓ Quel est la notation scientifique de \(0, 0000001101\)

Réponse

\(1,101 \times 2^{-7}\)

b. Représentation des nombres à virgule en binaire sur n bits⚓︎

Principe de la norme IEE754

On utilise la notation scientifique en binaire \((-1)^s \times 1,m \times 2^{p+127}\).
Pour une représentation sur 32 bits (en simple précision),
- le bit de poids fort (à gauche) donne le signe 0 pour positif et 1 pour négatif
- Les 8 bits suivants pour exposant, l'exposant de la notation scientifique auquel on ajoute 127, pour avoir des valeurs uniquement positives
- les 23 bits suivant pour partie après la virgule de la mantisse

norme IEE754

❓ Trouver la représentation en norme IEE754 sur 32 bits de \((1011, 0111 101)_2\) soit \((11,4765625)_{10}\)

Réponse
  1. On transforme en notation scientifique : \(1,011 0111 101 * 2^3\)
  2. On ajoute l'exposant, ici 127 : \(1,011 0111 101 * 2^{130}\)
  3. On écrit l'exposant en base 2 : \(1,011 0111 101 * 2^{10000010}\)
  4. On pose notre bit de signe : \(0\)
  5. On isole notre exposant : 10000010
  6. Et nos 23 bits de mantisse : \(01101111010000000000000\)

❓ Trouver la représentation en norme IEE754 sur 32 bits de \((0, 0000001101)_2\) soit \((0.0126953125)_{10}\)

Réponse
  1. On transforme en notation scientifique : \(1,101 * 2^{-7}\)
  2. On ajoute l'exposant, ici 127 : \(1,101 * 2^{120}\)
  3. On écrit l'exposant en base 2 : \(1,101 * 2^{01111000}\)
  4. On pose notre bit de signe : \(0\)
  5. On isole notre exposant : 01111000
  6. Et nos 23 bits de mantisse : \(10100000000000000000000\)

❓ Trouver la représentation en binaire sur 32 bits de 1 01111110 1111000000000000000000000

Réponse
  1. le signe est négatif car \(s=1\)
  2. L'exposant et 01111110 soit 126. On retranche la précision de 127, on obtient \(-1\)
  3. La mantisse 1111 donne \(1,1111 * 2^{-1} = 0.11111\) cad \(1/2 + 1/4 +1/8+1/16+1/32=0.96875\)
  4. Soit au final, la valeur -0.96875

❓ Trouver la représentation en binaire sur 32 bits de 0 10000011 11100000000000000000000

Réponse
  1. le signe est négatif car \(s=0\)
  2. L'exposant et 10000011 soit 131. On retranche la précision de 127, on obtient \(4\)
  3. La mantisse 111 donne \(1,111 * 2^{4} = 11110\) cad 30

❓ Trouver la représentation en norme IEE754 sur 32 bits de \(128\)

réponse
  1. on code en binaire la partie entière : 10000000
  2. on passe en notation scientifique : \(1.0 * 2^{7}\)
  3. On ajoute l'exposant de précision : \(1.0 * 2^{7+127} = 1.0 * 2^{134}\)
  4. On code l'exposant en binaire : \(1.0 * 2^{10000110}\)
  5. le bit de signe est 0
  6. Le bit d'exposant est 10000110
  7. et les 23 bits de la mantisse sont 0000000000000000000000000

soit 0 10000110 0000000000000000000000000

❓ Trouver la représentation en norme IEE754 sur 32 bits de \(-32.75\)

réponse
  1. on code en binaire 100000.11
  2. on passe en notation scientifique : \(1.0000011 * 2^{5}\)
  3. On ajoute l'exposant de précision : \(1.0000011 * 2^{5+127} = 1.0000011 * 2^{132}\)
  4. On code l'exposant en binaire : \(1.0000011 * 2^{10000100}\)
  5. le bit de signe est 1, négatif
  6. Le bit d'exposant est 10000100
  7. et les 23 bits de la mantisse sont 00000110000000000000000

Convention

Par convention :
– Il y a deux zéro, un positif et un négatif : \(\pm 00000000 00000000000000000000000\)
– L’infini : \(\pm 11111111 00000000000000000000000\)
NaN : not a number : \(\pm 11111111 01000000000000000000000\)

3. Comment faire des tests d'egalité sur les flottants ? ❤⚓︎

Première réponse : ON N'EN FAIT PAS.

Si a et b sont deux flottants, le test classique

🐍 Script Python
if a == b :
    print("a et b sont égaux")

a de grandes chances d'échouer :

Le script

🐍 Script Python
1
2
3
4
5
6
a = 0.1
b = 0.3 - 0.2
if a == b :
    print("a et b sont égaux")
else :
    print("a et b sont différents")

renverra

📋 Texte
a et b sont différents

Si vraiment un test d'égalité est nécessaire, on ne va pas tester l'égalité entre a et b mais leur proximité, grâce à la valeur absolue de leur différence.

La fonction abs(a-b) renvoie un nombre positif égal à la distance entre a et b. Il faut alors décider d'un écart minimal e en dessous duquel on considèrera que a et b sont égaux.

Le script

🐍 Script Python
a = 0.1
b = 0.3-0.2
e = 10**(-12)
if abs(a-b) < e :
    print("a et b sont égaux")
else :
    print("a et b sont différents")

renverra

📋 Texte
a et b sont égaux

Par ailleurs la bibliothèque math de Python inclut à cet effet la fonction math.isclose().

🐍 Script Python
>>> from math import isclose
>>> isclose(0.2 + 0.1, 0.3)
True

un autre bug

Source : Formation ISN 2018 Lyon - Pascal Busac et Brice Portier - Licence CC BY NC SA

plateforme

23 août 1991, au large de la Norvège : premier tremblement de terre causé par un bug informatique : la base en béton destinée à supporter la plate-forme Sleipner a coulé.

Approximation de calcul dans le logiciel de dessin des ballasts.
⬇
Sous-estimation de 47% de l’épaisseur des parois des ballasts.
⬇
A 65m de profondeur, rupture d’un des ballasts.
⬇
La plate-forme (90 000 tonnes) coule et se brise à 220m de profondeur.
⬇
Séisme de magnitude 3 sur l’échelle de Richter.