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 :

implémentation de ls -R et allocation mémoire


Sujet :

C

Vue hybride

Message précédent Message précédent   Message suivant Message suivant
  1. #1
    Candidat au Club
    Inscrit en
    Juillet 2009
    Messages
    2
    Détails du profil
    Informations forums :
    Inscription : Juillet 2009
    Messages : 2
    Par défaut implémentation de ls -R et allocation mémoire
    Bonjour,
    Je suis entrain d'implémenter la fonction ls -R
    J'ai d'abord une fonction Liste qui (je crois) fonctionne bien...
    Puis une fonction ListeRecursive qui appelle ma fonction Liste
    j'ai plein d'erreur de ce type la quand je lance ma fonction ListeRecursive
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    ls(5056) malloc: *** error for object 0x1002d0: incorrect checksum for freed object - object was probably modified after being freed.
    *** set a breakpoint in malloc_error_break to debug
    Je sais que ma recursion fonctionne, car j'arrive bien a afficher la liste des sous répertoire, mais c'est quand j'appelle ma fonction Liste que je rencontre des problèmes.

    Alors je ne sais pas... quelles structures je dois allouer? Pourquoi? est-ce que je dois les "free"? Pourquoi? pfffff

    Please, reconciliez moi avec le C! D'avance merci

    Gabriel

    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
    int Liste (char* dirname){
    	DIR* dir;
    	struct dirent* dirent;
    	if (!(dir = opendir (dirname))){
    		perror("opendir");
    		return 0;
    	}
    	while((dirent = readdir(dir))){
    		if (strcmp(".",dirent->d_name) && strcmp("..",dirent->d_name))
    			printf("%s\n",dirent->d_name);
    	}
    	if (closedir(dir) != 0){
    		perror("closedir");
    		return 0;
    	}
    	return 1;
    }
    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
    27
    28
    29
    30
    31
    void ListeRecursive (char* dirname){
    	DIR* dir;
    	struct dirent* dirent;
    	char* dirname_tmp;
    	struct stat st;
    	printf("%s:\n",dirname);
    	if (Liste(dirname) == 0)
    		exit(EXIT_FAILURE);
    	if (!(dir = opendir (dirname))){
    		perror("opendir");
    		exit(EXIT_FAILURE);
    	}
    	while((dirent = readdir(dir))){
    		if (strcmp(".",dirent->d_name) && strcmp("..",dirent->d_name)){
    			if ((dirname_tmp = malloc(strlen(dirname)+strlen(dirent->d_name)+1)) == NULL){
    				perror("malloc");
    				exit(EXIT_FAILURE);
    			}
    			strcpy(dirname_tmp,dirname);
    			dirname_tmp = strcat(dirname_tmp,"/");
    			dirname_tmp = strcat(dirname_tmp,dirent->d_name);
    			if (stat(dirname_tmp,&st) == -1){
    				perror("stat");
    				exit(EXIT_FAILURE);
    			}
    			if (S_ISDIR(st.st_mode))
    				ListeRecursive(dirname_tmp);
    			free(dirname_tmp);
    		}
    	}
    	closedir(dir);

  2. #2
    Expert confirmé

    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
    Billets dans le blog
    2
    Par défaut
    déjà en prenant l'habitude de tester par rpport aux valeurs attendues :

    1. on n'écrit pas :

      Code : Sélectionner tout - Visualiser dans une fenêtre à part
      	if (!(dir = opendir (dirname)))
      mais

      Code : Sélectionner tout - Visualiser dans une fenêtre à part
      	if ((dir = opendir (dirname)) != NULL)
    2. on n'écrit pas :

      Code : Sélectionner tout - Visualiser dans une fenêtre à part
      	while((dirent = readdir(dir))){
      mais

      Code : Sélectionner tout - Visualiser dans une fenêtre à part
      	while((dirent = readdir(dir)) != NULL){
    3. On n'écrit pas :


      Code : Sélectionner tout - Visualiser dans une fenêtre à part
      if (strcmp(".",dirent->d_name) && strcmp("..",dirent->d_name))
      mais

      Code : Sélectionner tout - Visualiser dans une fenêtre à part
      1
      2
      if ((strcmp(".",dirent->d_name) != 0) && 
                    (strcmp("..",dirent->d_name) != 0))
    4. ...



    Déjà tu y verras (et le compilateur) plus clair... et tu auras des résultats plus corrects ..

    Ensuite on peut utiliser des instructions qui allègent :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    			strcpy(dirname_tmp,dirname);
    			dirname_tmp = strcat(dirname_tmp,"/");
    			dirname_tmp = strcat(dirname_tmp,dirent->d_name);
    est équivalent à :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
                 sprintf ( dirname_tmp,  '%s/%s', dirname, dirent->d-name);

  3. #3
    Expert confirmé
    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
    Par défaut
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    if ((dirname_tmp = malloc(strlen(dirname)+strlen(dirent->d_name)+1)) == NULL){
    ...
    }
    strcpy(dirname_tmp,dirname);
    dirname_tmp = strcat(dirname_tmp,"/");
    dirname_tmp = strcat(dirname_tmp,dirent->d_name);
    Le nombre de caractères alloués est insuffisant. Il doit être de
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    strlen(dirname)+strlen("/")+ strlen(dirent->d_name)+1
    (le +1 est pour le zéro terminal).

    soit
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    strlen(dirname)+strlen(dirent->d_name)+2

  4. #4
    Candidat au Club
    Inscrit en
    Juillet 2009
    Messages
    2
    Détails du profil
    Informations forums :
    Inscription : Juillet 2009
    Messages : 2
    Par défaut
    Citation Envoyé par souviron34 Voir le message
    déjà en prenant l'habitude de tester par rpport aux valeurs attendues :
    Ok ok je tacherais de prendre cette habitude..! Merci

    Citation Envoyé par diogene Voir le message
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    strlen(dirname)+strlen(dirent->d_name)+2
    Grrrrrr... Il est trop sévère gcc franchement... Je n'ai pas pu voir le bit de mémoire qu'il lui fallait!
    Je ne maîtrise pas trop gdb mais j'avais essayer de débugger un peu avec... je savais que c'était ce malloc qui merdait (en même temps c'était le seul).
    J'avais vu que les fonction strcpy et strcat ajoutait le \0 à la fin... mais j'ai pas pensé a lui allouer la mémoire.

    Merci beaucoup en tout cas, grâce à cette case mémoire en plus tout fonctionne!

  5. #5
    Inactif  
    Avatar de Mac LAK
    Profil pro
    Inscrit en
    Octobre 2004
    Messages
    3 893
    Détails du profil
    Informations personnelles :
    Âge : 50
    Localisation : France, Haute Garonne (Midi Pyrénées)

    Informations forums :
    Inscription : Octobre 2004
    Messages : 3 893
    Par défaut
    Citation Envoyé par gabrielc Voir le message
    J'avais vu que les fonction strcpy et strcat ajoutait le \0 à la fin... mais j'ai pas pensé a lui allouer la mémoire.
    Toujours penser qu'une chaîne occupe "strlen(...)+1" octets... Quoi qu'il arrive, y compris par strcpy, strcat, etc.
    Mac LAK.
    ___________________________________________________
    Ne prenez pas la vie trop au sérieux, de toutes façons, vous n'en sortirez pas vivant.

    Sources et composants Delphi sur mon site, L'antre du Lak.
    Pas de question technique par MP : posez-la dans un nouveau sujet, sur le forum adéquat.

    Rejoignez-nous sur : Serveur de fichiers [NAS] Le Tableau de bord projets Le groupe de travail ICMO

  6. #6
    Expert éminent
    Avatar de Médinoc
    Homme Profil pro
    Développeur informatique
    Inscrit en
    Septembre 2005
    Messages
    27 392
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 41
    Localisation : France

    Informations professionnelles :
    Activité : Développeur informatique
    Secteur : High Tech - Éditeur de logiciels

    Informations forums :
    Inscription : Septembre 2005
    Messages : 27 392
    Par défaut
    Citation Envoyé par souviron34 Voir le message
    on n'écrit pas :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    	if (!(dir = opendir (dirname)))
    mais

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    	if ((dir = opendir (dirname)) != NULL)
    Attention, là, tu changes le sens du programme. Une preuve de plus que la première écriture n'apporte que confusion.
    SVP, pas de questions techniques par MP. Surtout si je ne vous ai jamais parlé avant.

    "Aw, come on, who would be so stupid as to insert a cast to make an error go away without actually fixing the error?"
    Apparently everyone.
    -- Raymond Chen.
    Traduction obligatoire: "Oh, voyons, qui serait assez stupide pour mettre un cast pour faire disparaitre un message d'erreur sans vraiment corriger l'erreur?" - Apparemment, tout le monde. -- Raymond Chen.

  7. #7
    Expert confirmé

    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
    Billets dans le blog
    2
    Par défaut
    tu as bien raison .. Désolé

    Mais effectivement une preuve de plus...


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

Discussions similaires

  1. Pb d'allocation mémoire malloc
    Par oz80 dans le forum C++
    Réponses: 5
    Dernier message: 18/11/2005, 17h23
  2. Limite Allocation Mémoire d'un tableau d'entier
    Par l9ft b9hind dans le forum C++
    Réponses: 5
    Dernier message: 27/10/2005, 19h29
  3. Allocation mémoire
    Par DestyNov@ dans le forum C++
    Réponses: 9
    Dernier message: 23/08/2005, 08h09
  4. [Pointeur] Allocation mémoire
    Par Rayek dans le forum Langage
    Réponses: 22
    Dernier message: 20/05/2005, 10h26
  5. Allocation mémoire dynamique
    Par ITISAR dans le forum VB 6 et antérieur
    Réponses: 6
    Dernier message: 21/01/2005, 09h59

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