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 :

Déclaration d'un pointeur dans une fonction avec malloc


Sujet :

C

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

    Informations forums :
    Inscription : Avril 2011
    Messages : 447
    Par défaut Déclaration d'un pointeur dans une fonction avec malloc
    Bonjour,

    Je voudrai savoir si déclarer un pointeur dans une fonction et en utilisant malloc peut-être correcte sachant que toutes variables déclarées dans une fonction est détruite à la fin.

    Est-ce que faire ceci est correct (code au plus simple):

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
     
    char *ptr maFonctionQuiDéclareUnPtr()
    {
        char *ptr = malloc(1000);
        ....
        return ptr;
    }

    Pourriez-vous me dire si c'est juste ?
    Même si ptr qui est déclaré dans la fonction, soit supprimé à la fin de celle-ci, du fait que malloc est été invoqué, alors l'adresse renvoyé par le return est toujours valide.

  2. #2
    Expert confirmé
    Avatar de gerald3d
    Homme Profil pro
    Conducteur de train
    Inscrit en
    Février 2008
    Messages
    2 308
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 55
    Localisation : France, Côte d'Or (Bourgogne)

    Informations professionnelles :
    Activité : Conducteur de train
    Secteur : Transports

    Informations forums :
    Inscription : Février 2008
    Messages : 2 308
    Billets dans le blog
    5
    Par défaut
    Bonjour.

    Effectivement toute variable déclarée dans une fonction est détruite lors de la fin d'exécution de cette dernière. Mais ici tu utilises malloc(); qui va allouer un bloc mémoire dans le tas. C'est à dire en dehors de toute fonction. Donc en retour tu vas obtenir le pointeur de ce bloc mémoire non détruit. Ta fonction est donc toute à fait correcte.

  3. #3
    Membre prolifique
    Avatar de Sve@r
    Homme Profil pro
    Ingénieur développement logiciels
    Inscrit en
    Février 2006
    Messages
    12 814
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Oise (Picardie)

    Informations professionnelles :
    Activité : Ingénieur développement logiciels
    Secteur : Aéronautique - Marine - Espace - Armement

    Informations forums :
    Inscription : Février 2006
    Messages : 12 814
    Billets dans le blog
    1
    Par défaut
    Bonjour
    Citation Envoyé par hbx360 Voir le message
    Même si ptr qui est déclaré dans la fonction, soit supprimé à la fin de celle-ci, du fait que malloc est été invoqué, alors l'adresse renvoyé par le return est toujours valide.
    Exact.
    La variable "ptr" est détruite effectivement... mais sa valeur (qui contient l'adresse de la zone allouée) est renvoyée à l'appelant qui lui peut la récupérer et la traiter.

    C'est exactement la même chose que dans cet exemple
    Code c : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    unsigned int carre(int n) {
        unsigned int r=n*n;
        return r;
    }
    Certes "r" est perdue, mais sa valeur elle a été renvoyée donc pas de souci (enfin vis à vis de la fonction car si l'appelant, lui, ne la récupère pas c'est son problème).

    En fait il ne faut pas confondre "variable perdue" et "valeur perdue". Comme le dit gerald3d, l'allocation se fait dans la zone mémoire appelée "tas (de m...)". Cette allocation est gérée entièrement par les fonctions de type alloc. Ensuite l'adresse de la zone allouée est renvoyée par malloc() et tu la récupères dans "ptr"... puis toi-même tu renvoies ensuite cette valeur. Donc c'est "passe à ton voisin" et tant que tu ne perds pas cette valeur tout va bien.

    D'ailleurs ta fonction a tellement peu de différence avec la fonction malloc (qui elle-aussi est une fonction comme une autre avec ses variables internes détruites elles-aussi à sa fermeture) qu'elle peut se contenter de renvoyer directement son retour. Toutefois le but de ta fonction n'est pas de déclarer un pointeur (ça ne veut rien dire) mais d'allouer une zone et renvoyer son adresse. Donc un petit changement de nom s'impose...

    Code c : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    char *ptr maFonctionQuiRetourneAdresseZoneAllouee() {
        return malloc(1000);
    }
    Mon Tutoriel sur la programmation «Python»
    Mon Tutoriel sur la programmation «Shell»
    Sinon il y en a pleins d'autres. N'oubliez pas non plus les différentes faq disponibles sur ce site
    Et on poste ses codes entre balises [code] et [/code]

  4. #4
    Membre éclairé
    Profil pro
    Inscrit en
    Avril 2011
    Messages
    447
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Avril 2011
    Messages : 447
    Par défaut
    D'accord merci pour vos réponses.

    Mais pourquoi dire :
    Citation Envoyé par Sve@r Voir le message
    Bonjour

    "tas (de m...)".

  5. #5
    Membre prolifique
    Avatar de Sve@r
    Homme Profil pro
    Ingénieur développement logiciels
    Inscrit en
    Février 2006
    Messages
    12 814
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Oise (Picardie)

    Informations professionnelles :
    Activité : Ingénieur développement logiciels
    Secteur : Aéronautique - Marine - Espace - Armement

    Informations forums :
    Inscription : Février 2006
    Messages : 12 814
    Billets dans le blog
    1
    Par défaut
    Citation Envoyé par hbx360 Voir le message
    Mais pourquoi dire :
    Comme ça tu n'oublieras jamais le nom de cette zone mémoire : le tas.
    Mon Tutoriel sur la programmation «Python»
    Mon Tutoriel sur la programmation «Shell»
    Sinon il y en a pleins d'autres. N'oubliez pas non plus les différentes faq disponibles sur ce site
    Et on poste ses codes entre balises [code] et [/code]

  6. #6
    Membre éclairé
    Profil pro
    Inscrit en
    Avril 2011
    Messages
    447
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Avril 2011
    Messages : 447
    Par défaut
    Ah d'accord

  7. #7
    Expert confirmé
    Avatar de gerald3d
    Homme Profil pro
    Conducteur de train
    Inscrit en
    Février 2008
    Messages
    2 308
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 55
    Localisation : France, Côte d'Or (Bourgogne)

    Informations professionnelles :
    Activité : Conducteur de train
    Secteur : Transports

    Informations forums :
    Inscription : Février 2008
    Messages : 2 308
    Billets dans le blog
    5
    Par défaut
    Histoire d’être complet dans la terminologie la mémoire allouée pour une fonction s’appelle la pile.

    En générale cette pile fait 1Mo. Elle peut être différente selon ta machine. Il est important de savoir qu’elle a une taille finie. Si tu déclares un tableau par exemple dont la taille est supérieure à la pile tu auras droit à un joli plantage de ton application 😁.

  8. #8
    Membre émérite
    Homme Profil pro
    Chef de projet NTIC
    Inscrit en
    Juillet 2020
    Messages
    352
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 52
    Localisation : France, Moselle (Lorraine)

    Informations professionnelles :
    Activité : Chef de projet NTIC

    Informations forums :
    Inscription : Juillet 2020
    Messages : 352
    Par défaut
    Tas (heap) et pile (stack) sont les deux zones mémoires les plus citées pour les plateformes usuelles en C. Il faut garder en mémoire (pun intended) qu'il existe également deux autres zones importantes : la zone où sont allouées les variables globales (zone en read/write) qui initialisées au lancement du programme contrairement aux variables des autres zones, ainsi qu'un zone en read only qui permet de stocker les littéraux (les chaînes littérales par exemple ou d'autres constantes mais pas en C, sauf si on le fait explicitement en utilisant des extensions des compilos).
    À ces zones s'ajoutent aussi les portions mémoires allouées aux bibliothèques dynamiques, au kernel et toute autres zones spécifiées lors du link time (il y en a pour les symboles de debug) … bref il y en a toute une floppée suivant l'os, la plateforme, …
    On peut limite parler également des caches processeur voire des registres, mais ces derniers ne rentrent en général pas dans la description d'un modèle mémoire à la C.

    Sous Linux, un programme quelconque peut toujours avoir accès lisiblement à sa map mémoire via le fichier virtuel /proc/self/map ; par exemple avec cat
    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
    26
    $ cat /proc/self/maps 
    555f82ffb000-555f82ffd000 r--p 00000000 08:12 6818382                    /usr/bin/cat
    555f82ffd000-555f83002000 r-xp 00002000 08:12 6818382                    /usr/bin/cat
    555f83002000-555f83005000 r--p 00007000 08:12 6818382                    /usr/bin/cat
    555f83005000-555f83006000 r--p 00009000 08:12 6818382                    /usr/bin/cat
    555f83006000-555f83007000 rw-p 0000a000 08:12 6818382                    /usr/bin/cat
    555f84e0a000-555f84e2b000 rw-p 00000000 00:00 0                          [heap]
    7f207aec6000-7f207b1b0000 r--p 00000000 08:12 6833710                    /usr/lib/locale/locale-archive
    7f207b1b0000-7f207b1b3000 rw-p 00000000 00:00 0 
    7f207b1b3000-7f207b1df000 r--p 00000000 08:12 6819206                    /usr/lib/libc.so.6
    7f207b1df000-7f207b355000 r-xp 0002c000 08:12 6819206                    /usr/lib/libc.so.6
    7f207b355000-7f207b3a9000 r--p 001a2000 08:12 6819206                    /usr/lib/libc.so.6
    7f207b3a9000-7f207b3aa000 ---p 001f6000 08:12 6819206                    /usr/lib/libc.so.6
    7f207b3aa000-7f207b3ad000 r--p 001f6000 08:12 6819206                    /usr/lib/libc.so.6
    7f207b3ad000-7f207b3b0000 rw-p 001f9000 08:12 6819206                    /usr/lib/libc.so.6
    7f207b3b0000-7f207b3bf000 rw-p 00000000 00:00 0 
    7f207b3c1000-7f207b3e3000 rw-p 00000000 00:00 0 
    7f207b3e3000-7f207b3e5000 r--p 00000000 08:12 6819194                    /usr/lib/ld-linux-x86-64.so.2
    7f207b3e5000-7f207b40c000 r-xp 00002000 08:12 6819194                    /usr/lib/ld-linux-x86-64.so.2
    7f207b40c000-7f207b417000 r--p 00029000 08:12 6819194                    /usr/lib/ld-linux-x86-64.so.2
    7f207b418000-7f207b41a000 r--p 00034000 08:12 6819194                    /usr/lib/ld-linux-x86-64.so.2
    7f207b41a000-7f207b41c000 rw-p 00036000 08:12 6819194                    /usr/lib/ld-linux-x86-64.so.2
    7ffe0d2a4000-7ffe0d2c5000 rw-p 00000000 00:00 0                          [stack]
    7ffe0d369000-7ffe0d36d000 r--p 00000000 00:00 0                          [vvar]
    7ffe0d36d000-7ffe0d36f000 r-xp 00000000 00:00 0                          [vdso]
    ffffffffff600000-ffffffffff601000 --xp 00000000 00:00 0                  [vsyscall]

  9. #9
    Membre éclairé
    Profil pro
    Inscrit en
    Avril 2011
    Messages
    447
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Avril 2011
    Messages : 447
    Par défaut
    D'accord merci pour vos explications.

    @WhiteCrow aurais-tu éventuellement un lien pour des cours sur les emplacements mémoires que tu cite dans ton dernier message ?

    Sinon je regarderai sur internet.

  10. #10
    Membre émérite
    Homme Profil pro
    Chef de projet NTIC
    Inscrit en
    Juillet 2020
    Messages
    352
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 52
    Localisation : France, Moselle (Lorraine)

    Informations professionnelles :
    Activité : Chef de projet NTIC

    Informations forums :
    Inscription : Juillet 2020
    Messages : 352
    Par défaut
    Ce ne sont pas des choses que l'on retrouve dans des cours, du moins pas à ma connaissance. Ce sont des infos que tu retrouves dans des articles, des descriptions et manuels techniques, etc. Les mots clés sont C memory model, et ce que j'ai décrit convient surtout pour des architectures usuelles ; tu auras aussi des infos dans les manuels de gcc et clang, voire des linkers comme ld ou gold …

  11. #11
    Membre éclairé
    Profil pro
    Inscrit en
    Avril 2011
    Messages
    447
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Avril 2011
    Messages : 447
    Par défaut
    Merci pour ta réponse.

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

Discussions similaires

  1. Modification de pointeurs dans une fonction
    Par bluemartini dans le forum Débuter
    Réponses: 4
    Dernier message: 26/09/2008, 10h23
  2. Modifier le contenu de pointeurs dans une fonction
    Par Sol_Invictus dans le forum Débuter
    Réponses: 6
    Dernier message: 11/09/2008, 23h30
  3. Accès à listview dans une fonction avec borland builder 5
    Par xasmxasm dans le forum C++Builder
    Réponses: 4
    Dernier message: 13/05/2008, 20h16
  4. Pointeur dans une fonction ?
    Par sliiim6184 dans le forum C
    Réponses: 4
    Dernier message: 28/12/2006, 11h32
  5. [Turbo Pascal] Allocation et désallocation de pointeurs dans une fonction
    Par neird dans le forum Turbo Pascal
    Réponses: 13
    Dernier message: 17/11/2002, 20h14

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