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

Langage Delphi Discussion :

SetLength: libérer la mémoire après usage


Sujet :

Langage Delphi

  1. #1
    Membre éclairé
    Inscrit en
    Juin 2005
    Messages
    644
    Détails du profil
    Informations professionnelles :
    Secteur : Industrie

    Informations forums :
    Inscription : Juin 2005
    Messages : 644
    Points : 754
    Points
    754
    Par défaut SetLength: libérer la mémoire après usage
    si on utilise la stratégie 'setlength' on alloue facilement à discrétion de la mémoire 'typée' que l'on accroît facilement sans perdre la partie déjà allouée en accroissant le nombre d'élément.
    Comment dé-allouer correctement cette mémoire?
    J'utilise
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
     
    SetLength(V,0);
    V:=nil;
    sans avoir vraiment vu si cela est OK si ce n'est par sources non certifiées.

    Actuellement dans le doute j'utilise plutôt l'approche suivante qui est un peu + lourde du point de vue de la syntaxe mais pour laquelle je suis sur de déallouer la mémoire.
    Par exemple:
    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
     
    const M=100; 
    var p : pointer;
    var d : ^double;
    getmem(p,M*sizeof(double)); 
     
    ReallocMem( P,newsize); // pour changer dynamiquement la taille et restituer au passage la mémoire si newsize < oldsize
     
    d:=p;
    for i:=0 to M-1 do
       begin
       d^:=....
       inc(d);
       end;
    ...
     
    freemem(p);  // mémoire désallouée
    p:=nil; // eventuel
    ...

  2. #2
    Expert éminent
    Avatar de Qwazerty
    Homme Profil pro
    La très haute tension :D
    Inscrit en
    Avril 2002
    Messages
    3 906
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 44
    Localisation : France

    Informations professionnelles :
    Activité : La très haute tension :D
    Secteur : Service public

    Informations forums :
    Inscription : Avril 2002
    Messages : 3 906
    Points : 8 539
    Points
    8 539
    Par défaut
    Salut
    Apres quelques recherches, sauf erreur de ma part, tu peux tout simplement utiliser [Edit]
    Apres relecture de l'aide
    Citation Envoyé par Aide Finalize
    Si la variable spécifiée lors d'un appel à Finalize ne contient pas de variants, de chaînes longues ni d'interfaces, le compilateur élimine l'appel et ne génère pas de code pour cet appel.
    Dans ce cas, il semblerait qu'il faille faire appelle à Dispose.

    En écrivant tout ceci, je me rend compte que je ne manipule pas souvent des emplacement mémoire, quelqu'un va certainement éclairer tout ça, parce qu'il faut tenir compte de ce qu'est V pour pouvoir te répondre a mon avis.

    [/Edit]

    ++
    Qwaz

  3. #3
    Expert éminent sénior
    Avatar de Paul TOTH
    Homme Profil pro
    Freelance
    Inscrit en
    Novembre 2002
    Messages
    8 964
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 55
    Localisation : France, Paris (Île de France)

    Informations professionnelles :
    Activité : Freelance
    Secteur : High Tech - Éditeur de logiciels

    Informations forums :
    Inscription : Novembre 2002
    Messages : 8 964
    Points : 28 457
    Points
    28 457
    Par défaut
    alors bien souvent tu n'as pas à libérer ladite mémoire si c'est une variable locale à une fonction, le tableau sera automatiquement libéré en fin de fonction.

    ensuite SetLength(x, 0) et x := nil sont équivalent, la seconde forme invoque directement DynArrayClear tandis que la première le fait en deux temps...

  4. #4
    Expert éminent
    Avatar de Qwazerty
    Homme Profil pro
    La très haute tension :D
    Inscrit en
    Avril 2002
    Messages
    3 906
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 44
    Localisation : France

    Informations professionnelles :
    Activité : La très haute tension :D
    Secteur : Service public

    Informations forums :
    Inscription : Avril 2002
    Messages : 3 906
    Points : 8 539
    Points
    8 539
    Par défaut
    Salut
    Désolé de poser mes baguages dans ce post, mais qu'en est il de Dispose et Finalize?
    ++
    Qwaz

  5. #5
    Membre éclairé
    Inscrit en
    Juin 2005
    Messages
    644
    Détails du profil
    Informations professionnelles :
    Secteur : Industrie

    Informations forums :
    Inscription : Juin 2005
    Messages : 644
    Points : 754
    Points
    754
    Par défaut
    Dispose est le pendant de l'opérateur 'New'

    êtes-vous sur que faire simplement

    V:=nil;

    libère la mémoire?

    J'ai des souvenirs (lointains) qu'avec TurboPascal sous DOS, mettre p:= nil perdait l'adresse mémoire mais ne désallouait pas une mémoire assignées pas getmem et qu'un appel à freemem était nécessaire. Mais j'ai pas ou très peu utilisé 'SetLength'.

  6. #6
    Expert éminent sénior
    Avatar de Paul TOTH
    Homme Profil pro
    Freelance
    Inscrit en
    Novembre 2002
    Messages
    8 964
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 55
    Localisation : France, Paris (Île de France)

    Informations professionnelles :
    Activité : Freelance
    Secteur : High Tech - Éditeur de logiciels

    Informations forums :
    Inscription : Novembre 2002
    Messages : 8 964
    Points : 28 457
    Points
    28 457
    Par défaut
    Citation Envoyé par Qwazerty Voir le message
    Salut
    Désolé de poser mes baguages dans ce post, mais qu'en est il de Dispose et Finalize?
    ++
    Qwaz
    Dispose() est le pendant de New()

    FreeMem() est le pendant de GetMem()

    Finalize() est appelé automatiquement pour les tableaux dynamiques et par Dispose()
    il n'a besoin d'être utilisé que dans le cas d'une libération par FreeMem() d'un espace mémoire qui contiendrait des tableaux dynamiques ou des chaînes longues.

    exemple
    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
     
    type
      TMonRecord = record
        Str: string;
      end;
      TDeuxRecord = array[0..1] of TMonRecord;
     
    var
      R1: array of TMonRecord;
      R2: ^TDeuxRecord;
      R3: ^TMonRecord;
      R4: ^TMonRecord;
    begin
      SetLength(R1, 2);
      R1[0].Str := 'Hello';
      R1[1].Str := 'Hello';
      R1 := nil; // rien à faire de particulier
     
      New(R2);
      R2[0].Str := 'Hello';
      R2[1].Str := 'Hello';
      Dispose(R2); // rien à faire de particulier
     
      GetMem(R3, 2 * SizeOf(TMonRecord));
      R3.Str := 'Hello';
      R4 := R3;
      Inc(R4); // Second TMonRecord pointé par R3
      R4.Str := 'Hello';
      Finalize(R3, 2); // Libère les deux Str car personne ne le fera pour vous
      FreeMem(R3);
    end;

  7. #7
    Expert éminent sénior

    Avatar de sjrd
    Homme Profil pro
    Directeur de projet
    Inscrit en
    Juin 2004
    Messages
    4 517
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 34
    Localisation : Suisse

    Informations professionnelles :
    Activité : Directeur de projet
    Secteur : Enseignement

    Informations forums :
    Inscription : Juin 2004
    Messages : 4 517
    Points : 10 154
    Points
    10 154
    Par défaut
    Pour compléter, j'ajouterais que :
    est équivalent à
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    GetMem(P, SizeOf(P^));
    Initialize(P); // en gros met à 0 si nécessaire
    Et
    est équivalent à
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    Finalize(P^);
    FreeMem(P);
    De plus, pour les variables non pointeurs, Initialize est appelée automatiquement en début de méthode, et Finalize en fin de méthode dans un try..finally implicite.

    Ainsi, si j'ai une variable de type TStringDynArray :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    var
      Arr: TStringDynArray;
    begin
      SetLength(Arr, 3);
      // bla bla
    end;
    Le compilateur transforme ça tout seul en
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    var
      Arr: TStringDynArray;
    begin
      Initialize(Arr);
      try
        SetLength(Arr, 3);
        // bla bla
      finally
        Finalize(Arr);
      end;
    end;
    Donc pour les tableaux dynamiques, il ne faut tout simplement pas s'occuper de la gestion de la mémoire : le langage le prend en charge tout seul.
    C'est tout l'intérêt d'ailleurs, sinon on retourne aux bons vieux tableaux statiques alloués dynamiquement.

  8. #8
    Expert éminent sénior

    Avatar de sjrd
    Homme Profil pro
    Directeur de projet
    Inscrit en
    Juin 2004
    Messages
    4 517
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 34
    Localisation : Suisse

    Informations professionnelles :
    Activité : Directeur de projet
    Secteur : Enseignement

    Informations forums :
    Inscription : Juin 2004
    Messages : 4 517
    Points : 10 154
    Points
    10 154
    Par défaut
    Et attention le dernier code de Paul Toth est buggué :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
      GetMem(R3, 2 * SizeOf(TMonRecord));
      R3.Str := 'Hello';
      R4 := R3;
      Inc(R4); // Second TMonRecord pointé par R3
      R4.Str := 'Hello';
      Finalize(R3, 2); // Libère les deux Str car personne ne le fera pour vous
      FreeMem(R3);
    Il manque un Initialize :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
      GetMem(R3, 2 * SizeOf(TMonRecord));
      Initialize(R3^, 2);

  9. #9
    Expert éminent
    Avatar de Qwazerty
    Homme Profil pro
    La très haute tension :D
    Inscrit en
    Avril 2002
    Messages
    3 906
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 44
    Localisation : France

    Informations professionnelles :
    Activité : La très haute tension :D
    Secteur : Service public

    Informations forums :
    Inscription : Avril 2002
    Messages : 3 906
    Points : 8 539
    Points
    8 539
    Par défaut
    SAlut
    Et donc si j'ai bien suivi

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    Finalize(R3, 2); // Libère les deux Str car personne ne le fera pour vous
    devrait donc être

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    Finalize(R3^, 2); // Libère les deux Str car personne ne le fera pour vous

    Mais puisque R3 est déclaré en ^TMonRecord, est ce qu'on devrait plutôt pas garder
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    Finalize(R3, 2); // Libère les deux Str car personne ne le fera pour vous
    et donc du coup avoir
    au lieu de

    ++
    Qwaz

  10. #10
    Expert éminent sénior
    Avatar de Paul TOTH
    Homme Profil pro
    Freelance
    Inscrit en
    Novembre 2002
    Messages
    8 964
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 55
    Localisation : France, Paris (Île de France)

    Informations professionnelles :
    Activité : Freelance
    Secteur : High Tech - Éditeur de logiciels

    Informations forums :
    Inscription : Novembre 2002
    Messages : 8 964
    Points : 28 457
    Points
    28 457
    Par défaut
    non, sjrd a parfaitement raison, voici le code corrigé

    j'ai ajouté un type PMonRecord car "R4 := R3" ne passe pas sinon

    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
    32
    33
    34
     
    type
      TMonRecord = record
        Str: string;
      end;
      TDeuxRecord = array[0..1] of TMonRecord;
      PMonRecord = ^TMonRecord;
     
    procedure test;
    var
      R1: array of TMonRecord;
      R2: ^TDeuxRecord;
      R3: PMonRecord;
      R4: PMonRecord;
    begin
      SetLength(R1, 2);
      R1[0].Str := 'Hello';
      R1[1].Str := 'Hello';
      R1 := nil; // rien à faire de particulier
     
      New(R2);
      R2[0].Str := 'Hello';
      R2[1].Str := 'Hello';
      Dispose(R2); // rien à faire de particulier
     
      GetMem(R3, 2 * SizeOf(TMonRecord));
      Initialize(R3^, 2);
      R3.Str := 'Hello';
      R4 := R3;
      Inc(R4); // Second TMonRecord pointé par R3
      R4.Str := 'Hello';
      Finalize(R3^, 2); // Libère les deux Str car personne ne le fera pour vous
      FreeMem(R3);
    end;

Discussions similaires

  1. Libérer mémoire après un new
    Par walloey dans le forum C#
    Réponses: 6
    Dernier message: 17/02/2012, 15h35
  2. [dll] libérer une dll apres utilisation
    Par polo54 dans le forum API standards et tierces
    Réponses: 12
    Dernier message: 11/07/2009, 22h48
  3. Libérer la mémoire aprés utilisation de collection
    Par harris_macken dans le forum Collection et Stream
    Réponses: 7
    Dernier message: 27/03/2008, 00h01
  4. libération de la mémoire après traitement ?
    Par isachat666 dans le forum MS SQL Server
    Réponses: 2
    Dernier message: 07/12/2005, 19h29
  5. Comment libérer la mémoire d'un TList ?
    Par Tchaill39 dans le forum Langage
    Réponses: 8
    Dernier message: 16/11/2005, 17h53

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