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 :

Petit calcul avec décalages


Sujet :

C

  1. #1
    Membre éclairé
    Profil pro
    Inscrit en
    Avril 2006
    Messages
    266
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Avril 2006
    Messages : 266
    Par défaut Petit calcul avec décalages
    Bonjour,
    voilà mon code :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
        unsigned short var;
        unsigned char tab[2]={0x12, 0x34};
    Je n'arrive pas à récupérer dans ma variable var la valeur 1234.
    J'ai testé avec des décalages mais rien de bon.
    Puis j'ai testé comme ceci :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
        unsigned short var;
        unsigned char tab[2]={0x12, 0x34};
        unsigned char tmp;
        tmp = tab[0]*0x10*0x10+tab[1];
        printf("0x%08X\n", tmp);
    Mais ça m'affiche 34, ça a l'air de tronquer mon 12...
    Pouvez-vous m'aider ?

  2. #2
    Membre éclairé
    Profil pro
    Inscrit en
    Avril 2006
    Messages
    266
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Avril 2006
    Messages : 266
    Par défaut
    Citation Envoyé par sone47
    Bonjour,
    tu peux concatener deux chaines et faire ensuite un atoi de la partie qui t'interesse.
    Je fais ça comment ? si ce n'est pas trop demander...

  3. #3
    Rédacteur
    Avatar de Franck.H
    Homme Profil pro
    Développeur .NET
    Inscrit en
    Janvier 2004
    Messages
    6 951
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 48
    Localisation : France, Haut Rhin (Alsace)

    Informations professionnelles :
    Activité : Développeur .NET
    Secteur : Service public

    Informations forums :
    Inscription : Janvier 2004
    Messages : 6 951
    Par défaut
    Citation Envoyé par frizou11
    Je fais ça comment ? si ce n'est pas trop demander...
    Tu fait un sprintf dans le style:
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    char var [5];
    ...
    sprintf (var, "%d%d", tab[0], tab[1]);
    puis une conversion de var avec strtol devrait suffir.
    Mon Site
    Ma bibliothèque de gestion des chaînes de caractères en C

    L'imagination est plus importante que le savoir. A. Einstein

    Je ne répond à aucune question technique par MP, merci d'avance !

  4. #4
    Membre émérite
    Profil pro
    Inscrit en
    Mars 2005
    Messages
    865
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Mars 2005
    Messages : 865
    Par défaut
    En fait, tu y étais presque.
    Mais si tu ajoutes le résultat de ton calcul dans un char, bien sûr qu'il sera tronqué. Mets le dans ton short et ça marchera mieux.

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
     
      unsigned short var;
      unsigned char tab[2]={0x12, 0x34};
      var = tab[0]*0x10*0x10+tab[1];
      printf("0x%08X\n", var);
    Un unsigned char tient sur un octet. La valeur maximale d'un unsigned char est donc 0xFF. Si tu calcules un nombre plus grand que 0xFF, il sera alors tronqué. C'est ce qui se passait dans ton cas.

    Tu peux aussi faire plus efficace en décalant tes bits.

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
     
      unsigned short var;
      unsigned char tab[2]={0x12, 0x34};
      var = ((short)tab[0] << 8) + tab[1];
      printf("0x%08X\n", var);

  5. #5
    Membre éclairé
    Profil pro
    Inscrit en
    Avril 2006
    Messages
    266
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Avril 2006
    Messages : 266
    Par défaut
    Ok, merci beaucoup aoyou

  6. #6
    Membre chevronné Avatar de straasha
    Homme Profil pro
    Développeur en systèmes embarqués
    Inscrit en
    Juillet 2004
    Messages
    149
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 45
    Localisation : France, Haute Garonne (Midi Pyrénées)

    Informations professionnelles :
    Activité : Développeur en systèmes embarqués
    Secteur : Transports

    Informations forums :
    Inscription : Juillet 2004
    Messages : 149
    Par défaut
    je me demande si
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    var = *(unsigned short*)tab;
    ne marche pas ?

  7. #7
    Membre éclairé
    Profil pro
    Inscrit en
    Avril 2006
    Messages
    266
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Avril 2006
    Messages : 266
    Par défaut
    Citation Envoyé par straasha
    je me demande si
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    var = *(unsigned short*)tab;
    ne marche pas ?
    Ca donne le résultat suivant :
    Et non 1234. Pas loin ! Je comprends pas pourquoi d'ailleurs...

    C'est comme ce morceau de code là :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
        unsigned short var;
        unsigned char tab[2]={0x12, 0x34};
        memcpy(&var, tab, 2);
        printf("0x%08X\n", var);
    Je comprends pas non plus pourquoi ça donne 3412 et non 1234

    Si quelqu'un voit pourquoi, ça serait bien de me l'indiquer...

  8. #8
    Membre chevronné Avatar de straasha
    Homme Profil pro
    Développeur en systèmes embarqués
    Inscrit en
    Juillet 2004
    Messages
    149
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 45
    Localisation : France, Haute Garonne (Midi Pyrénées)

    Informations professionnelles :
    Activité : Développeur en systèmes embarqués
    Secteur : Transports

    Informations forums :
    Inscription : Juillet 2004
    Messages : 149
    Par défaut
    arg ! tant pis... j'aurais tenté

  9. #9
    Membre émérite
    Profil pro
    Inscrit en
    Mars 2005
    Messages
    865
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Mars 2005
    Messages : 865
    Par défaut
    Je comprends pas non plus pourquoi ça donne 3412 et non 1234.
    Les processeurs x86 (ce qui doit être ton cas) sont en Little Endian, c'est-à-dire que les bits de poids faibles ont l'adresse mémoire la plus faible. Les bits de poids forts sont en dernier peut-on dire aussi. Donc quand tu fais ton memcpy, tu copies 12 à 0x01 pas exemple et 34 à 0x02 (valeurs d'adresse absurde mais c'est pour l'exemple). Quand tu demandes de lire un short, il récupère les bits de poids fort (c'est 34) et les bits de poids faible (c'est 12) d'où 3412 comme résultat et non 1234.

    Si tu avais travaillé sur un ancien MAC (avant qu'ils passent en Intel), tu aurais bien eu 1234 car ce sont des processeurs Big Endian.

    Si tu veux faire du code portable, il ne faut jamais préjuger de l'endianess. Tu as suffisamment d'opérateurs disponibles en C pour éviter de faire ces magouilles sur la mémoire.

  10. #10
    Membre éclairé
    Profil pro
    Inscrit en
    Avril 2006
    Messages
    266
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Avril 2006
    Messages : 266
    Par défaut
    Citation Envoyé par aoyou
    Les processeurs x86 (ce qui doit être ton cas) sont en Little Endian, c'est-à-dire que les bits de poids faibles ont l'adresse mémoire la plus faible. Les bits de poids forts sont en dernier peut-on dire aussi. Donc quand tu fais ton memcpy, tu copies 12 à 0x01 pas exemple et 34 à 0x02 (valeurs d'adresse absurde mais c'est pour l'exemple). Quand tu demandes de lire un short, il récupère les bits de poids fort (c'est 34) et les bits de poids faible (c'est 12) d'où 3412 comme résultat et non 1234.

    Si tu avais travaillé sur un ancien MAC (avant qu'ils passent en Intel), tu aurais bien eu 1234 car ce sont des processeurs Big Endian.

    Si tu veux faire du code portable, il ne faut jamais préjuger de l'endianess. Tu as suffisamment d'opérateurs disponibles en C pour éviter de faire ces magouilles sur la mémoire.
    Alors je dis "très bonne explication". Merci à toi, j'ai tout pigé
    Je savais qu'il y avait une histoire de Little Endian et Big Endian...

+ Répondre à la discussion
Cette discussion est résolue.

Discussions similaires

  1. [Débutant] Petit souci avec calcul
    Par Attila54 dans le forum VB.NET
    Réponses: 31
    Dernier message: 11/10/2011, 08h49
  2. Réponses: 11
    Dernier message: 19/09/2010, 14h37
  3. [DEBUTANT] petits soucis avec un prgm de chat
    Par LechucK dans le forum MFC
    Réponses: 8
    Dernier message: 19/01/2004, 16h52
  4. [debutant] pour debbuger un petit prog avec menu
    Par niluge01 dans le forum Windows
    Réponses: 3
    Dernier message: 22/11/2003, 14h03

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