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 C++ Discussion :

Factory générique avec/sans variadic templates


Sujet :

Langage C++

Vue hybride

Message précédent Message précédent   Message suivant Message suivant
  1. #1
    Expert confirmé

    Avatar de dragonjoker59
    Homme Profil pro
    Software Developer
    Inscrit en
    Juin 2005
    Messages
    2 033
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 43
    Localisation : France, Bas Rhin (Alsace)

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

    Informations forums :
    Inscription : Juin 2005
    Messages : 2 033
    Billets dans le blog
    12
    Par défaut Factory générique avec/sans variadic templates
    Bonjour!

    Tout d'abord, le 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
    25
    26
    27
    28
    29
    30
    31
    32
    template< class Obj, class Key, typename Creator >
    class Factory
    {
    protected:
    	typedef std::shared_ptr< Obj > ObjPtr;
    	typedef std::map< Key, Creator > ObjMap;
     
    public:
    	Factory() {}
    	virtual ~Factory() {}
    	void Register( Key const & p_key, Creator p_creator )
    	{
    		m_registered.insert( std::make_pair( p_key, p_creator ) );
    	}
    #if HAS_VARIADIC_TEMPLATES
    	template< typename ... Parameters >
    	ObjPtr Create( Key const & p_key, Parameters && ... p_params )
    	{
    		ObjPtr l_return;
    		typename ObjMap::iterator l_it = m_registered.find( p_key );
    		if ( l_it != m_registered.end() )
    		{
    			l_return = l_it->second( p_params... );
    		}
    		return l_return;
    	}
    #else
    #endif
     
    protected:
    	ObjMap m_registered;
    };
    Ce code fonctionne très bien sur un compilateur supportant les variadic templates, cependant j'aimerais savoir s'il y a possibilité de reproduire ce comportement avec un compilateur ne supportant pas les variadic templates, en gros, que puis-je mettre dans le #else?

    J'ai essayé avec une bidouille de fonctions activées via les std::enable_if, en créant une structure template permettant de déduire le nombre de paramètres que prend la fonction de création, mais sans succès.

    Quelqu'un a-t-il une idée?

    Merci d'avance!
    Si vous ne trouvez plus rien, cherchez autre chose...

    Vous trouverez ici des tutoriels OpenGL moderne.
    Mon moteur 3D: Castor 3D, presque utilisable (venez participer, il y a de la place)!
    Un projet qui ne sert à rien, mais qu'il est joli (des fois) : ProceduralGenerator (Génération procédurale d'images, et post-processing).

  2. #2
    Membre Expert

    Homme Profil pro
    Ingénieur développement logiciels
    Inscrit en
    Août 2004
    Messages
    1 391
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 35
    Localisation : France, Doubs (Franche Comté)

    Informations professionnelles :
    Activité : Ingénieur développement logiciels

    Informations forums :
    Inscription : Août 2004
    Messages : 1 391
    Par défaut
    Bonjour,

    Solution lourde mais très simple à comprendre : tu fais autant de version que tu veux gérer de paramètre :
    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
     
    template<class T1>
    	ObjPtr Create( Key const & p_key, T1&& p1)
    	{
    		ObjPtr l_return;
    		typename ObjMap::iterator l_it = m_registered.find( p_key );
    		if ( l_it != m_registered.end() )
    		{
    			l_return = l_it->second(p1);
    		}
    		return l_return;
    	}
     
    template<class T1, class T2>
    	ObjPtr Create( Key const & p_key, T1&& p1, T2&& p2)
    	{
    		ObjPtr l_return;
    		typename ObjMap::iterator l_it = m_registered.find( p_key );
    		if ( l_it != m_registered.end() )
    		{
    			l_return = l_it->second(p1,p2);
    		}
    		return l_return;
    	}
    Si tu peux utiliser boost, boost.PP devrait te permettre d'éviter d'écrire trop de code en utilisant le pré-compilateur pour faire les répétitions.

    NB: Tu as oublié des forward dans ton code.

  3. #3
    Expert confirmé

    Avatar de dragonjoker59
    Homme Profil pro
    Software Developer
    Inscrit en
    Juin 2005
    Messages
    2 033
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 43
    Localisation : France, Bas Rhin (Alsace)

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

    Informations forums :
    Inscription : Juin 2005
    Messages : 2 033
    Billets dans le blog
    12
    Par défaut
    Le problème est là : je ne sais pas combien de paramêtres le Creator va nécessiter.
    De plus à la déclaration d'une factory avec, par exemple, un Creator à 2 paramètres, le compilateur me gueule dessus à cause des autres fonctions qui essaient d'utiliser le Creator avec un nombre incorrect de paramètres.
    Pour les forward ça n'a l'air de poser de problèmes ni à VC12 ni à GCC 4.7 (Debian Wheezy amd64), ni à GCC 4.8.1 (MinGW).
    De plus je ne savais pas que l'on pouvait forwarder des paramètres variadiques comme ça...
    Si vous ne trouvez plus rien, cherchez autre chose...

    Vous trouverez ici des tutoriels OpenGL moderne.
    Mon moteur 3D: Castor 3D, presque utilisable (venez participer, il y a de la place)!
    Un projet qui ne sert à rien, mais qu'il est joli (des fois) : ProceduralGenerator (Génération procédurale d'images, et post-processing).

  4. #4
    Rédacteur/Modérateur


    Homme Profil pro
    Network game programmer
    Inscrit en
    Juin 2010
    Messages
    7 147
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 38
    Localisation : Canada

    Informations professionnelles :
    Activité : Network game programmer

    Informations forums :
    Inscription : Juin 2010
    Messages : 7 147
    Billets dans le blog
    4
    Par défaut
    tu as essayé avec les macro var_args ?
    Pensez à consulter la FAQ ou les cours et tutoriels de la section C++.
    Un peu de programmation réseau ?
    Aucune aide via MP ne sera dispensée. Merci d'utiliser les forums prévus à cet effet.

  5. #5
    Membre Expert

    Homme Profil pro
    Ingénieur développement logiciels
    Inscrit en
    Août 2004
    Messages
    1 391
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 35
    Localisation : France, Doubs (Franche Comté)

    Informations professionnelles :
    Activité : Ingénieur développement logiciels

    Informations forums :
    Inscription : Août 2004
    Messages : 1 391
    Par défaut
    L'idée est de te fixer un nombre maximal de paramètre N, et d'écrire N+1 versions, c'est juste long à écrire, mais ça fonctionne parfaitement.

    Par exemple ce foncteur avec des variadiques :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
     
    struct A
    {
      template<class... Arg>
      void operator()(Arg&&... arg) const
      {
        foo(std::forward<Arg>(arg)...);
      }
    };
    Donne sans les variadiques :
    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
     
    struct A
    {
      void operator()() const
      {
        foo();
      }
      template<class T>
      void operator()(T&& arg) const
      {
        foo(std::forward<T>(t));
      }
      template<class T1, class T2>
      void operator()(T1&& t1, T2&& t2) const
      {
        foo(std::forward<T1>(t1),std::forward<T2>(t2));
      }
      template<class T1, class T2, class T3>
      void operator()(T1&& t1, T2&& t2, T3&& t3) const
      {
        foo(std::forward<T1>(t1),std::forward<T2>(t2),std::forward<T3>(t3));
      }
    };
    Là je gère au maximum 3 paramètres. Après boost.pp pour rendre ça plus simple à écrire, et dans ce cas le N est défini par une macro ce qui permet à un utilisateur de le changer si il a besoin de gérer un nombre maximal de paramètre plus important. (Code non testé, mais c'est l'idée à appliquer)

    Pour ton problème de compilateur qui te stop à cause de fonction qui sont syntaxiquement invalides, c'est là qu'il faut utiliser le SFINAE / std::enable_if pour ne pas générer ces fonctions (ce n'est pas systématiquement nécessaire, on est dans une situation qui impliquent pas mal de template, ceux-ci ne sont générés que si ils sont utilisées, donc si ils doivent être syntaxiquement valide).

    PS: Pour les forward le compilateur ne te dit rien car ton code est parfaitement valide, par contre il ne profite pas du perfect forwarding.

  6. #6
    Expert éminent
    Avatar de koala01
    Homme Profil pro
    aucun
    Inscrit en
    Octobre 2004
    Messages
    11 644
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 53
    Localisation : Belgique

    Informations professionnelles :
    Activité : aucun

    Informations forums :
    Inscription : Octobre 2004
    Messages : 11 644
    Par défaut
    Salut,

    As-tu déjà jeté un oeil au code que je propose dans cette discussion Il pourrait peut être t'aider à trouver une solution

    Perso, j'ai simplement décidé d'utiliser des tuples pour maintenir la liste des paramètres requis par le constructeur. Mais le code de test indique tout cela...
    A méditer: La solution la plus simple est toujours la moins compliquée
    Ce qui se conçoit bien s'énonce clairement, et les mots pour le dire vous viennent aisément. Nicolas Boileau
    Compiler Gcc sous windows avec MinGW
    Coder efficacement en C++ : dans les bacs le 17 février 2014
    mon tout nouveau blog

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

Discussions similaires

  1. Réponses: 4
    Dernier message: 30/05/2011, 19h38
  2. Classe Liste générique avec template
    Par TNT89 dans le forum Langage
    Réponses: 9
    Dernier message: 09/05/2009, 15h14
  3. zend framework avec ou sans moteur templates ?
    Par yveson33 dans le forum Zend Framework
    Réponses: 3
    Dernier message: 12/03/2008, 21h25
  4. Faire une recherche avec/sans accents
    Par Wedge3D dans le forum MS SQL Server
    Réponses: 3
    Dernier message: 21/02/2007, 10h43
  5. Appeler une fonction avec/sans parenthèses
    Par haypo dans le forum Algorithmes et structures de données
    Réponses: 8
    Dernier message: 29/12/2002, 18h48

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