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 :

sizeof() et taille d'un buffer passé en paramètres de fonction


Sujet :

C

  1. #1
    Rédacteur

    Avatar de ram-0000
    Homme Profil pro
    Consultant en sécurité
    Inscrit en
    Mai 2007
    Messages
    11 517
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 61
    Localisation : France, Haute Garonne (Midi Pyrénées)

    Informations professionnelles :
    Activité : Consultant en sécurité
    Secteur : High Tech - Opérateur de télécommunications

    Informations forums :
    Inscription : Mai 2007
    Messages : 11 517
    Points : 50 369
    Points
    50 369
    Par défaut sizeof() et taille d'un buffer passé en paramètres de fonction
    Bonjour,

    J'ai un petit problème de compréhension d'un comportement avec le code suivant :
    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
    void ma_fct1(char Param[6])
    {
    	printf("ma_fct1(), size = %d\n", sizeof(Param));
    }
     
    void ma_fct2(char * Param)
    {
    	printf("ma_fct2(), size = %d\n", sizeof(Param));
    }
     
    void ma_fct3(char Param[8])
    {
    	printf("ma_fct3(), size = %d\n", sizeof(Param));
    }
     
    int main(int /*argc*/, char * /*argv*/ [])
    {
    	char buff[6];
    	printf("main(), size = %d\n", sizeof(buff));
    	ma_fct1(buff);
    	ma_fct2(buff);
    	ma_fct3(buff);
     
    	return 0;
    }
    Le résultat généré est le suivant :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    main(), size = 6
    ma_fct1(), size = 4
    ma_fct2(), size = 4
    ma_fct3(), size = 4
    La taille du buffer est correctement calculée par sizeof() dans le main() mais n'est plus le résultat attendu dans les différentes fonctions (ma_fct1, ma_fct2 et ma_fct3). C'est comme ci, malgré la définition de la fonction, le compilateur disait que ce n'est pas un buffer de 6 ou 8 caractères mais un pointeur sur caractères (ce qui est vrai aussi).

    D'autant plus que le compilateur ne hurle pas lorsque l'on passe un buffer de 6 à une fonction qui s'attend à recevoir un buffer de 8.

    Quelqu'un aura t'il une explication ?

  2. #2
    Modérateur
    Avatar de Obsidian
    Homme Profil pro
    Développeur en systèmes embarqués
    Inscrit en
    Septembre 2007
    Messages
    7 403
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 48
    Localisation : France, Essonne (Île de France)

    Informations professionnelles :
    Activité : Développeur en systèmes embarqués
    Secteur : High Tech - Éditeur de logiciels

    Informations forums :
    Inscription : Septembre 2007
    Messages : 7 403
    Points : 23 787
    Points
    23 787
    Par défaut
    Citation Envoyé par ram-0000 Voir le message
    La taille du buffer est correctement calculée par sizeof() dans le main() mais n'est plus le résultat attendu dans les différentes fonctions (ma_fct1, ma_fct2 et ma_fct3). C'est comme ci, malgré la définition de la fonction, le compilateur disait que ce n'est pas un buffer de 6 ou 8 caractères mais un pointeur sur caractères (ce qui est vrai aussi).
    C'est le cas.

    Autant que je sache, le C ne sait pas passer directement les tableaux en tant qu'argument de fonction. Donc, tant que le tableau est directement visible, ça va. Par contre, si tu passes un tableau à une fonction, ce sera obligatoirement à travers un pointeur. Et comme, de fait, ce pointeur existe et est déclaré en mémoire en tant que tel, c'est bien à lui que sizeof va se référer.

    La seule astuce que je connaisse pour « tricher » est de déclarer un tableau (de taille fixe) en tant que membre d'une structure. Passer cette structure permet donc de transmettre le contenu du tableau qu'elle contient.

    Maintenant, j'ai toujours péché sur les conversions pointeurs↔tableau. Il y a probablement plus érudit que moi pour te répondre…

  3. #3
    Membre éprouvé Avatar de orfix
    Homme Profil pro
    Inscrit en
    Avril 2007
    Messages
    707
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : Maroc

    Informations professionnelles :
    Secteur : High Tech - Produits et services télécom et Internet

    Informations forums :
    Inscription : Avril 2007
    Messages : 707
    Points : 1 132
    Points
    1 132
    Par défaut
    Tout devient clair à partir du moment où l'on sait que le nom d'un tableau est implicitement converti en pointeur vers son premier élément (sauf devant sizeof et &), autrement dit on aurait pu écrire:
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    	ma_fct1(&buff[0]);
    	ma_fct2(&buff[0]);
    	ma_fct3(&buff[0]);
    Vous voyez donc bien qu'écrire:
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    void ma_fct1(char Param[6])
    void ma_fct3(char Param[8])
    /* Voir même */
    void ma_fct4(char Param[99999])
    Ne change absolument rien, le compilateur voit cela toujours comme:
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    void ma_fct1(char *Param)
    /* Ou */
    void ma_fct1(char Param[])

  4. #4
    Expert éminent
    Avatar de Melem
    Homme Profil pro
    Ingénieur développement logiciels
    Inscrit en
    Janvier 2006
    Messages
    3 656
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 38
    Localisation : France, Essonne (Île de France)

    Informations professionnelles :
    Activité : Ingénieur développement logiciels
    Secteur : High Tech - Électronique et micro-électronique

    Informations forums :
    Inscription : Janvier 2006
    Messages : 3 656
    Points : 8 389
    Points
    8 389
    Par défaut
    ram-0000 : En effet dans la spécification du type d'un argument d'une fonction, les déclarations char *, char [] et char [N] (N entier quelconque) sont strictement équivalentes, et vallent char *. Il en est ainsi car de toute façon, lorsque tu passes un tableau en argument d'une fonction, le tableau est systématiquement remplacé par l'adresse de son premier élément. D'ailleurs, les seuls cas où une expression de type tableau n'est pas remplacé par l'adresse de son premier élément s'est lorsqu'il est en unique argument de sizeof (sizeof(t)), en unique argument de l'opérateur "adresse de" (&t) ou en unique argument à droite de l'opérateur d'initialisation (const char * = "...").

  5. #5
    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
    On peut signaler un cas supplémentaire où le tableau reste un tableau, c'est en tant qu'expression parenthèsée : si t est un tableau, (t) est un tableau. C'est d'ailleurs ce qui t'a permis d'écrire sizeof(t) aussi bien que sizeof t.
    Naturellement, il ne s'agit pas ici des délimiteurs ( et ) qui encadrent la liste des arguments d'une fonction.

  6. #6
    Membre éclairé
    Avatar de D[r]eadLock
    Profil pro
    Inscrit en
    Mai 2002
    Messages
    504
    Détails du profil
    Informations personnelles :
    Âge : 45
    Localisation : France

    Informations forums :
    Inscription : Mai 2002
    Messages : 504
    Points : 750
    Points
    750
    Par défaut
    En résumé, voici la regle que j'utilise: n'utiliser [] que:
    1. dans les déclarations de variable et le plus souvent sans taille:
    2. dans les déclaration de struct:
      Code : Sélectionner tout - Visualiser dans une fenêtre à part
      struct foo { int array[SIZE]; char name[MAX_NAME_SIZE]...};


    Et donc jamais dans des paramètres de fonctions, uniquement des pointeurs, et je n'ai jamais eu de soucis

  7. #7
    Rédacteur

    Avatar de ram-0000
    Homme Profil pro
    Consultant en sécurité
    Inscrit en
    Mai 2007
    Messages
    11 517
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 61
    Localisation : France, Haute Garonne (Midi Pyrénées)

    Informations professionnelles :
    Activité : Consultant en sécurité
    Secteur : High Tech - Opérateur de télécommunications

    Informations forums :
    Inscription : Mai 2007
    Messages : 11 517
    Points : 50 369
    Points
    50 369
    Par défaut
    Donc si je résume,
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    void ma_fct1(char Param[6])
    est automatiquement transformé en interne par le compilateur en
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    void ma_fct1(char * Param)
    et donc ce type de déclaration ou de définition ne présente d'intérêt que pour le développeur/relecteur. Il indique explicitement ce qu'il s'attend à trouver sachant que le compilateur le traite légèrement différemment.

  8. #8
    Expert éminent sénior

    Profil pro
    Inscrit en
    Janvier 2007
    Messages
    10 610
    Détails du profil
    Informations personnelles :
    Âge : 67
    Localisation : France

    Informations forums :
    Inscription : Janvier 2007
    Messages : 10 610
    Points : 17 923
    Points
    17 923
    Billets dans le blog
    2
    Par défaut
    je +1 avec D[r]eadLock

  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
    Citation Envoyé par ram-0000 Voir le message
    Donc si je résume,
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    void ma_fct1(char Param[6])
    est automatiquement transformé en interne par le compilateur en
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    void ma_fct1(char * Param)
    et donc ce type de déclaration ou de définition ne présente d'intérêt que pour le développeur/relecteur. Il indique explicitement ce qu'il s'attend à trouver sachant que le compilateur le traite légèrement différemment.
    Oui.
    Ceci est traité exactement de la même façon par le compilateur
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    void ma_fct1(char Param[6])
    void ma_fct1(char Param[])
    void ma_fct1(char *Param)
    et dans les trois cas, Param est du type char*.

    Personnellement, je n'aime pas la première forme qui introduit une grandeur (6) qui n'a pas de signification. La troisième forme est la plus conforme au type de Param. Mais, on peut préférer selon le cas une des deux premières formes à la troisième mais alors dans un but purement documentaire.

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

Discussions similaires

  1. Réponses: 6
    Dernier message: 04/12/2012, 20h04
  2. Réponses: 11
    Dernier message: 01/02/2011, 16h20
  3. Réponses: 13
    Dernier message: 21/03/2009, 21h49
  4. Nom d'une variable passée en paramètre à une fonction
    Par cata2 dans le forum Interfaces Graphiques
    Réponses: 1
    Dernier message: 07/09/2007, 10h17
  5. Réponses: 13
    Dernier message: 07/05/2006, 12h54

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