IdentifiantMot de passe
Loading...
Mot de passe oublié ?Je m'inscris ! (gratuit)
Navigation

Inscrivez-vous gratuitement
pour pouvoir participer, suivre les réponses en temps réel, voter pour les messages, poser vos propres questions et recevoir la newsletter

C Discussion :

Addition sur des floats


Sujet :

C

  1. #1
    Invité
    Invité(e)
    Par défaut Addition sur des floats
    Bonjour,

    Je suis surpris d'une erreur d'addition dans le programme C ci dessous.

    Le but du programme est de trouver le nombre de fibonnaci tel que F(x) mod 2^32 == 0.

    Mon problème vient d'une addition de float particulière...

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    35 => 9227465 : mod = 9227465.0
    36 => 14930352 : mod = 14930352.0
    37 => 24157816 : mod = 24157816.0
    normalement a la position 37 nous devrions avoir 24157817.

    voici mon code :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    #include <stdio.h>
    #include <stdlib.h>
    #include <math.h>
     
    int main (void)
    {
        float f,x;
        float f1,f2;
        int stop = 0;
        float ans = pow(2,32);
        f1 = 0;
        f2 = 1;
        for(x=37;stop!=1;x++)
        {
            f = f1+f2;
            if (fmod(f,ans) == 0)
                stop =1;
     
            printf("%.0f => %.0f : mod = %.01f\n",x,f, fmod(f,ans));
            f2 = f1;
            f1 = f;
        }
        return 0;
     
    }

  2. #2
    Membre éclairé
    Avatar de Pouet_forever
    Profil pro
    Inscrit en
    Octobre 2009
    Messages
    671
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Octobre 2009
    Messages : 671
    Points : 842
    Points
    842
    Par défaut
    Ce qu'il faut savoir avec les nombres flottants c'est qu'ils font une approximation
    Plus tu pédales moins fort, moins t'avances plus vite.

  3. #3
    Invité
    Invité(e)
    Par défaut
    C'est dommage (pour mon cas, car sinon ça doit avoir une raison). Alors comment faire pour passer outre cette approximation ?

  4. #4
    Membre émérite
    Profil pro
    Inscrit en
    Octobre 2008
    Messages
    1 874
    Détails du profil
    Informations personnelles :
    Localisation : France, Paris (Île de France)

    Informations forums :
    Inscription : Octobre 2008
    Messages : 1 874
    Points : 2 890
    Points
    2 890
    Par défaut
    Pour avoir une meilleure précision, tu peux utiliser le type double au lieu du float. D'autant plus que les fonctions pow() et modf() sont définies pour prendre en argument et renvoyer des double et pas des float.
    D'autre part le code que tu montres commençe à x=37, donc on ne peut pas reproduire tel quel le résultat que tu montres.

  5. #5
    Expert confirmé

    Inscrit en
    Août 2006
    Messages
    3 945
    Détails du profil
    Informations forums :
    Inscription : Août 2006
    Messages : 3 945
    Points : 5 659
    Points
    5 659
    Par défaut
    Soa,

    Même le type double posera des problèmes d'approximations, c'est intrinsèque à la représentation des réels.

    Si tu veux des calculs exacts, il faut utiliser des entiers. Et comme les valeurs que tu cherches dépassent les limites des types standards, il faut utiliser une bibliothèque de calcul en multi-précision. Il y a entre autres GMP qui marche très bien.
    Si les cons volaient, il ferait nuit à midi.

  6. #6
    Invité
    Invité(e)
    Par défaut
    Citation Envoyé par estofilo Voir le message
    Pour avoir une meilleure précision, tu peux utiliser le type double au lieu du float. D'autant plus que les fonctions pow() et modf() sont définies pour prendre en argument et renvoyer des double et pas des float.
    D'autre part le code que tu montres commençe à x=37, donc on ne peut pas reproduire tel quel le résultat que tu montres.
    Pour le départ à 37 c'est une erreur de ma part dans le collage du code sur le forum, on peut partir à 1 tout simplement en remplaçant x=37 par x=1 dans la boucle for.

    Citation Envoyé par droggo Voir le message
    Soa,

    Même le type double posera des problèmes d'approximations, c'est intrinsèque à la représentation des réels.

    Si tu veux des calculs exacts, il faut utiliser des entiers. Et comme les valeurs que tu cherches dépassent les limites des types standards, il faut utiliser une bibliothèque de calcul en multi-précision. Il y a entre autres GMP qui marche très bien.
    Je pensais en effet à GMP mais je voulais voir s'il n'y avait pas d'autres solutions avant d'utiliser cela. Et bien tant pis. Merci à tous. Je vais tester GMP.

  7. #7
    Expert éminent sénior
    Avatar de diogene
    Homme Profil pro
    Enseignant Chercheur
    Inscrit en
    Juin 2005
    Messages
    5 761
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Essonne (Île de France)

    Informations professionnelles :
    Activité : Enseignant Chercheur
    Secteur : Enseignement

    Informations forums :
    Inscription : Juin 2005
    Messages : 5 761
    Points : 13 926
    Points
    13 926
    Par défaut
    Le but du programme est de trouver le nombre de fibonnaci tel que F(x) mod 2^32 == 0.
    Il n'y a pas besoin de calculer 2^32 pour cela, ni d'utiliser des float ou des double.

    On a : F(x) = F(x-1) + F(x-2)
    alors : F(x)mod(n) = ( F(x-1)mod(n) + F(x-2) mod(n) ) mod(n)
    avec n = 2^32

    Comme l'arithmétique des nombres entiers unsigned est une arithmétique mod 2^p, ou 2^p - 1 est la plus grande valeur représentable par ce type, il suffit de représenter F(x) par un nombre d'un type entier unsigned possédant 32 bits (ou plus), par exemple unsigned long, et d'ignorer purement et simplement pendant le calcul les dépassements de capacité au delà des 32 bits.

    Pour illustrer, supposons que n = 16 = 2^4 et F(x) sur 4 bits :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
     
    x      F(x)  F(x) mod 16
    .....
     6      8      8   (1000)
     7     13     13   (1101)
     8     21      5 = (8+13) mod(16)  1000+1101 = (1)0101 
     9     34      2 = (5+13) mod(16)  1101+0101 = (1)0010
    10     55      7 = (5+2)  mod(16)  0101+0010 =    0111
    11     89      9 = (2+7)  mod(16)  0010+0111 =    1001
    12    144      0 = (7+9)  mod(16)  0111+1001 = (1)0000
    .....
    F(3 221 225 472) == 0 mod(2^32)
    Publication : Concepts en C

    Mon avatar : Glenn Gould

    --------------------------------------------------------------------------
    Une réponse vous a été utile ? Remerciez son auteur en cliquant le pouce vert !

  8. #8
    Membre émérite
    Profil pro
    Inscrit en
    Octobre 2008
    Messages
    1 874
    Détails du profil
    Informations personnelles :
    Localisation : France, Paris (Île de France)

    Informations forums :
    Inscription : Octobre 2008
    Messages : 1 874
    Points : 2 890
    Points
    2 890
    Par défaut
    En suivant la méthode de diogene, il me semble que la solution de F(x) mod 2^32 == 0 est un x qui est de l'ordre de 3 milliards, ce qui doit donner un F(x) absolument gigantesque, qui dépasse ce qui est gérable par les types natifs du langage C.

    Il doit être possible de recouper le résultat avec la méthode générique de calculer les vraies valeurs de F(x) et le reste de la division à chaque tour de boucle en utilisant GMP, mais ça risque de prendre un temps de calcul considérable.

  9. #9
    Expert éminent sénior
    Avatar de diogene
    Homme Profil pro
    Enseignant Chercheur
    Inscrit en
    Juin 2005
    Messages
    5 761
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Essonne (Île de France)

    Informations professionnelles :
    Activité : Enseignant Chercheur
    Secteur : Enseignement

    Informations forums :
    Inscription : Juin 2005
    Messages : 5 761
    Points : 13 926
    Points
    13 926
    Par défaut
    La solution que j'ai obtenue était cachée dans mon message précédent
    Publication : Concepts en C

    Mon avatar : Glenn Gould

    --------------------------------------------------------------------------
    Une réponse vous a été utile ? Remerciez son auteur en cliquant le pouce vert !

  10. #10
    Invité
    Invité(e)
    Par défaut
    De retour de vacances j'ai pris note des messages.

    J'avoue ne pas avoir encore compris la réponse de diogène mais je m'y penche.

    Merci pour les explications.

Discussions similaires

  1. [SimpleXML] Opérations mathématiques sur des float
    Par CBresso dans le forum Bibliothèques et frameworks
    Réponses: 3
    Dernier message: 11/05/2012, 13h38
  2. codage des floats sur 4 bytes
    Par pfeuh dans le forum Débuter
    Réponses: 5
    Dernier message: 27/01/2010, 15h30
  3. Réponses: 9
    Dernier message: 17/06/2009, 15h10
  4. IE6 ajoute des espaces sur div float
    Par deejay2221 dans le forum Mise en page CSS
    Réponses: 8
    Dernier message: 15/08/2008, 11h25
  5. Test if sur des float
    Par Minuit dans le forum Linux
    Réponses: 2
    Dernier message: 26/03/2005, 13h08

Partager

Partager
  • Envoyer la discussion sur Viadeo
  • Envoyer la discussion sur Twitter
  • Envoyer la discussion sur Google
  • Envoyer la discussion sur Facebook
  • Envoyer la discussion sur Digg
  • Envoyer la discussion sur Delicious
  • Envoyer la discussion sur MySpace
  • Envoyer la discussion sur Yahoo