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

Bibliothèques Discussion :

[BOOST] shared_ptr et void*


Sujet :

Bibliothèques

Vue hybride

Message précédent Message précédent   Message suivant Message suivant
  1. #1
    Membre du Club
    Inscrit en
    Juin 2006
    Messages
    9
    Détails du profil
    Informations forums :
    Inscription : Juin 2006
    Messages : 9
    Par défaut [BOOST] shared_ptr et void*
    Bonjour,

    Je travaille sous Borland Developer Studio 2006 et je souhaite utiliser cette magnifique invention que sont les "smart pointers", et spécialement le shared_ptr de la librairie BOOST. Tout se passe bien dans mes classes métier mais je coince quand je souhaite peupler une TListView avec les objets contenus dans un std::vector< shared_ptr<...> >. Au moment de peupler mon TListView, je dois associer un objet à la propriété Data de chaque TListItem, et cet objet doit avoir un type générique "void*". Ensuite, quand je veux récupérer l'objet associé à un TListItem sur lequel l'utilisateur vient de cliquer, je ne sais pas comment procéder pour convertir de void* vers shared_ptr<...>.

    Exemple : j'ai un classe CPersonne qui décrit une personne (nom et prénom). J'ai un singleton qui permet de gérer une espèce de carnet d'adresses contenant des personnes. En gros, j'ai les donc les instructions suivantes dans mon code :
    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
     
    class CPersonne
    {
    private:
        std::string _Nom;
        std::string _Prenom;
     
    public:
    ...
    };
     
    typedef boost::shared_ptr<CPersonne> TPersonne;
    typedef std::vector<TPersonne> TListePersonnes;
     
    class CGestionnairePersonnes {
    private:
        TListePersonnes _ListePersonnes;
     
    public:
        unsigned int GetNbPersonnes() {return _ListePersonnes.size();}
     
        TPersonne GetPersonne(int i) {return _ListePersonnes.at(i);}
    ...
    }
    Quand je veux peupler la TListView avec les personnes du singleton, je fais :

    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
        ListViewPersonnes->Items->BeginUpdate();
    
        ListViewPersonnes->Items->Clear();
    
        for (unsigned int i = 0; i < PERSONNES.GetNbPersonnes(); i++)
        {
            TListItem* item = ListViewPersonnes->Items->Add();
            TPersonne personne = PERSONNES.GetPersonne(i);
            item->Data = boost::shared_ptr<void>(personne).get();
            item->Caption = personne->GetPrenom().c_str();
            item->SubItems->Add(personne->GetNom().c_str());
        }
    
        ListViewPersonnes->Items->EndUpdate();
    C'est l'instruction en rouge qui, selon moi, permet de convertir le shared_ptr en un void* acceptable par la TListView.

    Ensuite, je ne sais pas comment reconvertir ce void* en TPersonne pour pouvoir l'exploiter, par exemple lors de la sélection d'un TListItem :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
     
    void __fastcall TFormMain::ListViewPersonnesSelectItem(TObject *Sender,
          TListItem *Item, bool Selected)
    {
        TPersonne p = ???????? Item->Data ?????;
    }
    Quelqu'un a une idée ? A base de shared_ptr<void>, static_pointer_cast, ou tout autre chose ?

    Merci !

  2. #2
    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
    pour ce genre de truc, je serais plutôt du genre à créer un shared_ptr sur le tas (comme c'est un objet) et passer son adresse en paramètre...
    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.

  3. #3
    Membre du Club
    Inscrit en
    Juin 2006
    Messages
    9
    Détails du profil
    Informations forums :
    Inscription : Juin 2006
    Messages : 9
    Par défaut
    A quel moment ? A l'affectation dans le Data du TListItem ?

  4. #4
    Rédacteur

    Avatar de Matthieu Brucher
    Profil pro
    Développeur HPC
    Inscrit en
    Juillet 2005
    Messages
    9 810
    Détails du profil
    Informations personnelles :
    Âge : 43
    Localisation : France, Pyrénées Atlantiques (Aquitaine)

    Informations professionnelles :
    Activité : Développeur HPC
    Secteur : Industrie

    Informations forums :
    Inscription : Juillet 2005
    Messages : 9 810
    Par défaut
    Si tu as absolument besoin du void*, une solution est d'inclure le compteur de référence dans la classe - intrusive_ptr - ou encore mieux, de faire dériver ta classe de enable_shared_from_this, comme indiqué dans le tuto suivant : http://miles.developpez.com/tutoriel...ost/smartptrs/

  5. #5
    Membre du Club
    Inscrit en
    Juin 2006
    Messages
    9
    Détails du profil
    Informations forums :
    Inscription : Juin 2006
    Messages : 9
    Par défaut
    Miles : j'avais bien lu ton article avant de poser la question (c'est d'ailleurs lui qui m'a décidé à utiliser les smart pointeurs). Mais je ne vois pas très bien comment résoudre le problème en dérivant de enable_shared_from_this En reprenant ton exemple de gestionnaire de textures, comment ferais-tu pour remplir une TListView (avec leur nom ou leur emplacement sur le disque, peu importe...) ? Je souhaite évidemment conserver l'association d'un objet avec un TListItem...

    Merci de vos réponses en tout cas !

  6. #6
    Rédacteur

    Avatar de Matthieu Brucher
    Profil pro
    Développeur HPC
    Inscrit en
    Juillet 2005
    Messages
    9 810
    Détails du profil
    Informations personnelles :
    Âge : 43
    Localisation : France, Pyrénées Atlantiques (Aquitaine)

    Informations professionnelles :
    Activité : Développeur HPC
    Secteur : Industrie

    Informations forums :
    Inscription : Juillet 2005
    Messages : 9 810
    Par défaut
    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
    class CPersonne : public boost::enable_shared_from_this<CPersonne>
    {
    private:
        std::string _Nom;
        std::string _Prenom;
    
    public:
    ...
    };
    
    //...
        for (unsigned int i = 0; i < PERSONNES.GetNbPersonnes(); i++)
        {
            TListItem* item = ListViewPersonnes->Items->Add();
            TPersonne personne = PERSONNES.GetPersonne(i);
            item->Data = static_cast<void *>(personne.get()); // On ne stocke dans Data que le pointeur vers la bête
            item->Caption = personne->GetPrenom().c_str();
            item->SubItems->Add(personne->GetNom().c_str());
        }
    //...
        TPersonne p = (static_cast<CPersonne *>(Item->Data))->shared_from_this();
    Essaie qqch du genre
    En fait, tu ne stockes pas le pointeur intelligent mais le pointeur lui-même, puis tu recrées le pointeur intelligent car l'instance sait elle-même où est stocké le compteur de référence donc c'est toujours encore le même compteur qui est utilisé.

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

Discussions similaires

  1. Réponses: 8
    Dernier message: 25/03/2008, 21h59
  2. Réponses: 12
    Dernier message: 25/02/2008, 14h27
  3. Copie de boost::shared_ptr
    Par Kurisu dans le forum Boost
    Réponses: 2
    Dernier message: 07/09/2006, 15h29
  4. boost::shared_ptr et singletons
    Par Elendil_BzH dans le forum Bibliothèques
    Réponses: 2
    Dernier message: 15/01/2006, 20h45
  5. [BOOST] shared_ptr et pointer C
    Par zdra dans le forum Bibliothèques
    Réponses: 7
    Dernier message: 08/05/2005, 14h15

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