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 :

Pointeur sur méthode membre


Sujet :

C++

Vue hybride

Message précédent Message précédent   Message suivant Message suivant
  1. #1
    Membre confirmé
    Profil pro
    Étudiant
    Inscrit en
    Janvier 2008
    Messages
    56
    Détails du profil
    Informations personnelles :
    Âge : 38
    Localisation : France, Calvados (Basse Normandie)

    Informations professionnelles :
    Activité : Étudiant

    Informations forums :
    Inscription : Janvier 2008
    Messages : 56
    Par défaut Pointeur sur méthode membre
    Bonjour à tous.

    Voici ma classe qui me permet de charger une scène 3D depuis un fichier xml.
    J'utilise la bibliothèque libxml++ qui à chaque fois qu'elle rencontre un noeud récupère son nom ainsi que tous ses attributs.
    Son nom est sous forme de caractères donc impossible de switcher et hors de question de faire des if, ca deviendrait vite fait illisible. J'utilise donc un map avec le nom en clé, et un pointeur sur fonction en valeur, dont le rôle est de traiter un noeud xml en fonction de son nom.
    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
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    57
    58
    59
    60
    61
    62
    63
    64
    65
    66
    67
    68
    69
     
    ////////////////////////////////////////////////////////////////
    // Cette classe permet de charger une scène 3D depuis un fichier xml
    ////////////////////////////////////////////////////////////////
    class _sd_Export_ SceneParser : public xmlpp::SaxParser
    {
    private:
     
    	/** 
             * Objet Root utilisé pour créer la scène.
             */
    	Ogre::Root* mRoot;
     
    	/**
             * Objet SceneManager utilisé pour créer des entitées...
             */
    	Ogre::SceneManager* mSceneManager;
     
    	/**
             * Node parent pour la création des nodes enfants présents dans le fichier xml.
             */
    	Ogre::SceneNode* mParentNode;
     
    	/** 
             * Liste des nodes.
             * @Note
             *              A chaque node que l'on rencontre on l'insert dans le conteneur. Si on rencontre la
             *              balise de fin de node on le supprime du conteneur car il n'aura pas d'autre node enfant.
             */
    	typedef std::stack<Ogre::SceneNode*> NodeStack;
    	NodeStack mNodesList;
     
    	/**
             * Type pointeur de méthode de traitement d'une balise.
             * @Note
             *              Chaque méthode traite une balise spécifique en fonction du type de la balise (son nom).
             */
    	typedef void (*ElementParser)(const AttributeList&s);
     
    	/** 
             * Ce map contient une chaine de caractère en clé (nom de la balise) associée à une méthode de traitement d'une balise.
             */
    	typedef std::map<Glib::ustring, ElementParser> ParserMap;
    	ParserMap mParserMap;
     
    public:
     
    	//......................................................
    	// Constructeur par défaut.
    	//......................................................
    	SceneParser(Ogre::Root* myRoot);
     
    	//......................................................
    	// Destructeur.
    	//......................................................
    	virtual ~SceneParser();
     
    protected:
     
    	//......................................................
    	// SaxParser::on_start_element().
    	//......................................................
    	virtual void on_start_element(const Glib::ustring& name, const AttributeList& attributes);
     
    	//......................................................
    	// SaxParser::on_end_element().
    	//......................................................
    	virtual void on_end_element(const Glib::ustring& name);
    };
    La méthode on_start_element est appelé à chaque début de noeud
    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
     
    void SceneParser::on_start_element(const Glib::ustring& name, const AttributeList& attributes)
    {
         Logger::Log() << "	Début de l'élément : " << name << ".\n"; 
     
         Logger::Log() << "		Nom du noeud : " << name << "\n";
     
         // On affiche les attributs du noeud courant.
         for(xmlpp::SaxParser::AttributeList::const_iterator Iter = attributes.begin(); Iter != attributes.end(); ++Iter)
         {
    	Logger::Log() << "			Attribut : " << Iter->name << " = " << Iter->value << "\n";
         }
     
         // Rechercher la fonction associée à l'élément.
         ParserMap::const_iterator It = mParserMap.find(name);
         if(It != mParserMap.end())
         {
             // On appelle la fonction associée.
            (It->second)(attributes);
         }
    }
    On voit donc que cette méthode est très courte et qu'elle délègue le travail aux fonctions approprié (préalablement enregistrée dans le map)

    Voici donc mes questions :
    Comment insérer des pointeurs sur fonctions membres dans mon map puisque ce ne sont pas des fonctions simples et comment déclarer ce map?

    PS : désolé Davidbrcz j'ai modifier au dernier moment, m'en veux pas ^^

  2. #2
    Rédacteur

    Avatar de Davidbrcz
    Homme Profil pro
    Ing Supaéro - Doctorant ONERA
    Inscrit en
    Juin 2006
    Messages
    2 307
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 33
    Localisation : Suisse

    Informations professionnelles :
    Activité : Ing Supaéro - Doctorant ONERA

    Informations forums :
    Inscription : Juin 2006
    Messages : 2 307
    Par défaut
    Il est où le problème ?
    "Never use brute force in fighting an exponential." (Andrei Alexandrescu)

    Mes articles dont Conseils divers sur le C++
    Une très bonne doc sur le C++ (en) Why linux is better (fr)

  3. #3
    Membre chevronné
    Avatar de NewbiZ
    Profil pro
    Étudiant
    Inscrit en
    Juillet 2002
    Messages
    184
    Détails du profil
    Informations personnelles :
    Âge : 40
    Localisation : France

    Informations professionnelles :
    Activité : Étudiant

    Informations forums :
    Inscription : Juillet 2002
    Messages : 184
    Par défaut
    J'ai pas trop regardé le source, mais vite fait :

    Solution 1:
    Déclaration de pointeur sur méthode : type0 (class::*func)(type1, type2, ...)

    Solution 2:
    Eliminer le problème de déclaration et déléguer le dispatch à une fonction statique :
    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
    class Foo
    {
      type0 func1(type1, type2) {}
      type0 func2(type1, type2) {}
     
      static type0 call_func1( Foo* p, type1 t1, type2 t2)
      {
        p->func1(t1,t2);
      }
     
      static type0 call_func2( Foo* p, type1 t1, type2 t2)
      {
        p->func2(t1,t2);
      }
    };

  4. #4
    Membre confirmé
    Profil pro
    Étudiant
    Inscrit en
    Janvier 2008
    Messages
    56
    Détails du profil
    Informations personnelles :
    Âge : 38
    Localisation : France, Calvados (Basse Normandie)

    Informations professionnelles :
    Activité : Étudiant

    Informations forums :
    Inscription : Janvier 2008
    Messages : 56
    Par défaut
    La solution 2 est quand même très lourde, j'ai essayé la numéro 1 et voici ce que j'obtiens :
    .hpp
    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
     
    class _sd_Export_ SceneParser : public xmlpp::SaxParser
    {
    private:
         /**
          * Type pointeur de méthode de traitement d'une balise.
          * @Note
          *         Chaque méthode traite une balise spécifique en fonction du type de la balise (son nom).
          */
          typedef void (*SceneParser::ElementParser)();
     
         /** 
          * Ce map contient une chaine de caractère en clé (nom de la balise) associée à une méthode de traitement d'une balise.
          */
         typedef std::map<Glib::ustring, ElementParser> ParserMap;
         ParserMap mParserMap;
     
         ...
    }
    .cpp
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
     
    SceneParser::SceneParser()
    {
         mParserMap["node"] = SceneParser::NodeParser();
    }
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
     
    error C2440: '=' : impossible de convertir de 'void' en 'void (__cdecl *)(void)'
    Etrange

  5. #5
    Membre Expert
    Avatar de coyotte507
    Profil pro
    Inscrit en
    Octobre 2006
    Messages
    1 327
    Détails du profil
    Informations personnelles :
    Âge : 35
    Localisation : France

    Informations forums :
    Inscription : Octobre 2006
    Messages : 1 327
    Par défaut
    Peut-être que sans les parenthèses...

  6. #6
    Membre confirmé
    Profil pro
    Étudiant
    Inscrit en
    Janvier 2008
    Messages
    56
    Détails du profil
    Informations personnelles :
    Âge : 38
    Localisation : France, Calvados (Basse Normandie)

    Informations professionnelles :
    Activité : Étudiant

    Informations forums :
    Inscription : Janvier 2008
    Messages : 56
    Par défaut
    Non j'ai enlevé les paranthèses dans le .cpp et voici ce que j'obtiens
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
     
    error C3867: &'SoftDream::SceneParser::NodeParser' : liste d'arguments manquante dans l'appel de fonction ; utilisez 'SoftDream::SceneParser::NodeParser' pour créer un pointeur vers membre
    error C2440: '=' : impossible de convertir de 'void (__thiscall SoftDream::SceneParser::* )(void)' en 'void (__cdecl *)(void)'
    1>        Aucun contexte dans lequel cette conversion est possible

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

Discussions similaires

  1. Pointeur sur le membre d'une matrice
    Par Linu6 dans le forum C
    Réponses: 7
    Dernier message: 13/01/2008, 20h40
  2. [POO] Pointeur sur fonction membre et héritage
    Par MrDuChnok dans le forum C++
    Réponses: 9
    Dernier message: 20/07/2006, 17h19
  3. Pointeur sur fonction membre avec parametre
    Par Glosialabolas dans le forum C++
    Réponses: 7
    Dernier message: 06/02/2006, 02h32
  4. Réponses: 10
    Dernier message: 03/02/2005, 13h09
  5. Réponses: 5
    Dernier message: 12/01/2005, 20h58

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