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 :

[Conception] Polymorphisme paramétrique ou d'inclusion ? Cas litigieux…


Sujet :

C++

Vue hybride

Message précédent Message précédent   Message suivant Message suivant
  1. #1
    Membre chevronné
    Avatar de Florian Goo
    Profil pro
    Inscrit en
    Septembre 2008
    Messages
    680
    Détails du profil
    Informations personnelles :
    Âge : 39
    Localisation : France, Paris (Île de France)

    Informations forums :
    Inscription : Septembre 2008
    Messages : 680
    Par défaut [Conception] Polymorphisme paramétrique ou d'inclusion ? Cas litigieux…
    Bonne rentrée, les C++iens !

    Je me remets sur mon projet et je suis face à un problème de conception. J'ai hésité à poster dans le forum conception général, mais tout ceci est je trouve assez propre au C++.

    Voilà le problème.
    Je souhaite modéliser les entités sémantiques du C++ : les namespaces, les classes, les fonctions et tout et tout. Restons-en à ces trois types d'entités.
    Je suis partagé entre deux solutions :
    1. mettre en place un pattern composite transparent (une interface scope que les classes namespace_, class_ et function vont implémenter d'une part et agréger d'autre part), en utilisant le polymorphisme d'inclusion ;
    2. ou faire une totale distinction entre ces trois classes, en utilisant le polymorphisme paramétrique.


    Le problème avec la solution 1, c'est qu'elle n'est pas type-safe. Si j'écris une interface scope contenant une fonction permettant d'ajouter un namespace, cela n'aura pas de sens pour les objets de type class_ et function.
    Après, avec un pattern Visitor, on peut toujours s'en sortir.

    Le problème avec la solution 2, c'est la nécessité d'émuler le polymorphisme d'inclusion à plusieurs endroits du code (grâce à des outils du style boost::variant), tels que :
    • la liste (présente dans chaque objet des trois classes) des nœuds enfants dans l'arborescence des scopes ;
    • le type de retour de certaines fonctions (qui peuvent renvoyer un objet d'un des trois types).

    Cette nécessité m'apparait comme symptomatique d'un mauvais choix de type de polymorphisme, mais ce n'est rien de plus qu'une sensation.

    Chacune des deux solutions ne fait que me pousser vers l'autre. Le choix est tellement cornélien que je préfère me tourner vers ceux d'entre vous qui ont plus d'expérience en conception. Que faites-vous dans ce genre de situation ? (Peut-être une solution 1,5 ?)

    Merci à vous !
    Cours : Initiation à CMake
    Projet : Scalpel, bibliothèque d'analyse de code source C++ (développement en cours)
    Ce message a été tapé avec un clavier en disposition bépo.

  2. #2
    Expert confirmé

    Inscrit en
    Novembre 2005
    Messages
    5 145
    Détails du profil
    Informations forums :
    Inscription : Novembre 2005
    Messages : 5 145
    Par défaut
    Citation Envoyé par Florian Goo Voir le message
    Le problème avec la solution 1, c'est qu'elle n'est pas type-safe. Si j'écris une interface scope contenant une fonction permettant d'ajouter un namespace, cela n'aura pas de sens pour les objets de type class_ et function.
    Ce genre de restrictions est plus accidentelle qu'essentielle. Je ne les enforcerais pas dans le systeme de type.

    En passant, ceci peut t'interesser.

  3. #3
    Membre chevronné
    Avatar de Florian Goo
    Profil pro
    Inscrit en
    Septembre 2008
    Messages
    680
    Détails du profil
    Informations personnelles :
    Âge : 39
    Localisation : France, Paris (Île de France)

    Informations forums :
    Inscription : Septembre 2008
    Messages : 680
    Par défaut
    Solution 1 sans hésitation, alors ?
    Cours : Initiation à CMake
    Projet : Scalpel, bibliothèque d'analyse de code source C++ (développement en cours)
    Ce message a été tapé avec un clavier en disposition bépo.

  4. #4
    Alp
    Alp est déconnecté
    Expert confirmé

    Avatar de Alp
    Homme Profil pro
    Inscrit en
    Juin 2005
    Messages
    8 575
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 36
    Localisation : France, Bouches du Rhône (Provence Alpes Côte d'Azur)

    Informations forums :
    Inscription : Juin 2005
    Messages : 8 575
    Par défaut
    Oui, a priori. Tu peux enforcer les règles de scoping dans ton mécanisme de dispatch, ce que tu peux aussi faire avec du polymorphisme paramétrique mais qui lui te pousse à utiliser boost::variant alors que le polymorphisme d'inclusion semble adapté quoi.

  5. #5
    Expert confirmé
    Homme Profil pro
    Ingénieur développement logiciels
    Inscrit en
    Décembre 2003
    Messages
    3 549
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Essonne (Île de France)

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

    Informations forums :
    Inscription : Décembre 2003
    Messages : 3 549
    Par défaut
    Personnellement, je ne vois pas en quoi la solution 1 est pertinente, puisqu'il faudra downcaster ou visiter.
    L'utilisation d'un variant a plus de sens, mais peut être plus lourde syntaxiquement.

  6. #6
    Membre chevronné
    Avatar de Florian Goo
    Profil pro
    Inscrit en
    Septembre 2008
    Messages
    680
    Détails du profil
    Informations personnelles :
    Âge : 39
    Localisation : France, Paris (Île de France)

    Informations forums :
    Inscription : Septembre 2008
    Messages : 680
    Par défaut
    Si vous-même êtes en désaccord sur ce point, ça me rassure sur mon propre niveau .
    Mais ça ne fait que me tirailler encore plus entre les deux possibilités…

    Si cela pouvait donner lieu à un débat, ce serait formidable.
    Mais à défaut (ou même en complément), auriez-vous de la littérature pertinente sur le sujet ?
    Cours : Initiation à CMake
    Projet : Scalpel, bibliothèque d'analyse de code source C++ (développement en cours)
    Ce message a été tapé avec un clavier en disposition bépo.

  7. #7
    Expert confirmé

    Inscrit en
    Novembre 2005
    Messages
    5 145
    Détails du profil
    Informations forums :
    Inscription : Novembre 2005
    Messages : 5 145
    Par défaut
    Citation Envoyé par Florian Goo Voir le message
    Solution 1 sans hésitation, alors ?
    Non, je n'ai pas reflechi au probleme assez pour avoir une preference entre les solutions que tu indiques ou pour une alternative que je te proposerais -- pour bien concevoir il faut savoir non seulement ce qui est modele mais aussi le contexte d'utilisation. Ca demande du temps que je n'ai pas.

    Simplement la raison que tu donnes contre elle ne me semble pas particulierement pertinente.

  8. #8
    Membre très actif Avatar de metagoto
    Profil pro
    Hobbyist programmateur
    Inscrit en
    Juin 2009
    Messages
    646
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations professionnelles :
    Activité : Hobbyist programmateur

    Informations forums :
    Inscription : Juin 2009
    Messages : 646
    Par défaut
    Citation Envoyé par Florian Goo Voir le message
    Le problème avec la solution 2, c'est la nécessité d'émuler le polymorphisme d'inclusion à plusieurs endroits du code (grâce à des outils du style boost::variant), tels que :
    • la liste (présente dans chaque objet des trois classes) des nœuds enfants dans l'arborescence des scopes ;
    • le type de retour de certaines fonctions (qui peuvent renvoyer un objet d'un des trois types).
    En me basant sur ces seuls points, j'ai l'impression qu'un boost variant recursif pourrait faire l'affaire. J'ose même espérer que cela soit plus simple qu'une solution dynamique au regard des inconvénients qui ont été énumérés.
    Dans le doute, autant privilégier une solution statique.
    (après, je dis ça, je dis rien... )

  9. #9
    Membre averti
    Profil pro
    Inscrit en
    Septembre 2004
    Messages
    30
    Détails du profil
    Informations personnelles :
    Localisation : Suisse

    Informations forums :
    Inscription : Septembre 2004
    Messages : 30
    Par défaut
    Personnellement je définirais le meta-schéma de manière à ce qu'il soit type-safe. Donc une variation du modèle composite où n'importe quoi ne peut pas être inclu dans n'importe quoi Style:

    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
     
    class Namable {
      string name;
      list<Namable&> getChildren() = 0;
      // Pas d'attribut liste, délégué aux enfants
    };
     
    class Method : public Namable { 
      Class& parent;
    };
     
    class Class : public Namable {
      list<Method&> methods;
    };
     
    class Function: public Namable { 
    };
     
    class NameSpace : public Namable {
      list<Class&> classes;
      list<Functions&> functions;
    };
    Si tu veux éviter d'avoir plusieurs listes par classe, tu peux toujours faire des interfaces de marquage (genre INameSpaceable = je peux le mettre dans un name space) mais je crois qu'un typage fort te donnera une meilleure sémantique (par exemple en offrant une méthode NameSpace::getClasses()).

  10. #10
    Membre chevronné
    Avatar de Florian Goo
    Profil pro
    Inscrit en
    Septembre 2008
    Messages
    680
    Détails du profil
    Informations personnelles :
    Âge : 39
    Localisation : France, Paris (Île de France)

    Informations forums :
    Inscription : Septembre 2008
    Messages : 680
    Par défaut
    Merci pour vos réponses, je fais rentrer tout ça dans la machine à mouliner .

    Finalement ça débouche peut-être sur une question plus simple : quand utiliser le pattern Visitor et quand utiliser boost::variant (ou autre) ?
    Cours : Initiation à CMake
    Projet : Scalpel, bibliothèque d'analyse de code source C++ (développement en cours)
    Ce message a été tapé avec un clavier en disposition bépo.

  11. #11
    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,

    Pourquoi ne pas passer par des type traits

    Tu aurais une classe de base scope proche de
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    class Scope
    {
        public:
            Scope(Scope const *, std::string const &);
            std::string const & name()const;
            std::string const entiereScopeName() const;
            virtual void addNamespace(Scope const &) = 0;
            virtual void addClass(Scope const &) = 0;
            virtual void addFunction(Scope const &) = 0;
        private:
            Scope * parent_;
            std::string name_;
            std::list<Scope&> children_;
    };
    et des traits de politique proche de
    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
    struct AllowedNamespace
    {
        static void check(){/* just does nothing while there are  allowed */}
    };
    struct ForbiddenNamespace
    {
        static void check(){throw NameSpaceForbidden();}
    };
    struct AllowedClass
    {
        static void check(){/* just does nothing while there are  allowed */}
    };
    struct ForbiddenClass
    {
        static void check(){throw ClassForbidden();}
    };
    struct AllowedFunction
    {
        static void check(){/* just does nothing while there are  allowed */}
    };
    struct ForbiddenFunction
    {
        static void check(){throw FunctionForbidden();}
    };
    /* il y en a peut etre d'autres*/
    et une définition de ta classe concrète proche de
    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 NameSpacePolicy,
                  class ClassPolicy,
                  class FunctionPolicy
                  /*, et toutes les autres  */
            >
    class TConcreteScope : public Scope
    {
     
            virtual void addNamespace(Scope const& s)
            {
                NamespacePolicy::check();
                children_.push_back(s);
            }
            virtual void addClass(Scope const &)
            {
                ClassPolicy::check();
                children_.push_back(s);
            }
            virtual void addFunction(Scope const &)
            {
                FunctionPolicy::check();
                children_.push_back(s);
            }
    };
    Après, tu n'a plus qu'à définir les alias de types qui vont bien:
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    typedef TConcreteScope<AllowedNamespace,
                           AllowedClass,
                           AllowedFunction> MyNameSpace;
    De cette manière, tu profite tout à la fois du polymorphisme dynamique et de la souplesse de définition des différents scopes réels

    Nota: tu pourrais remplacer le lancement d'une exception par le simple fait de ne pas déclarer la fonction check dans les ForbiddenXX... ainsi, la vérification se ferait à la compilation

    Ceci dit, c'est peut être justement ce que propose boost::any
    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

  12. #12
    Membre chevronné
    Avatar de Florian Goo
    Profil pro
    Inscrit en
    Septembre 2008
    Messages
    680
    Détails du profil
    Informations personnelles :
    Âge : 39
    Localisation : France, Paris (Île de France)

    Informations forums :
    Inscription : Septembre 2008
    Messages : 680
    Par défaut
    Bon, concentrons-nous sur un cas d'utilisation en particulier, histoire d'ajouter les contraintes une à une. (Après, quel que soit la solution sur ce point, cela ne veut pas dire que je ne doive jouer qu'avec un seul type de polymorphisme, comme le suggère koala.)

    Je vais avoir des fonctions de recherche qui vont implémenter le name-lookup.

    Par exemple, il y a une fonction find_unqualified_name() à qui on donne un nom et un scope et qui nous renvoie l'entité associée à ce nom. Ça peut être une variable, une classe, un type en général, une fonction, un namespace… tout et n'importe quoi, inconnu à la compilation.

    Cette fonction devra donc renvoyer au choix :
    1. un named_entity* (où named_entity est une interface qui sera implémentée par les classes namespace_, class_, function, variable et toutes les autres) ;
    2. un boost::variant<namespace_, class_, function, variable, …>*.


    La solution 1 est simple, facile, classique. Je dirais même Java-like, avec les bons et les mauvais côtés.

    La solution 2 me parait peu envisageable, pas vous ?
    Certes, il sera toujours possible de rendre le code de la fonction modulaire et indépendant : le type de retour pourrait être passé en paramètre template. On pourrait imaginer un header définissant un typedef de ce variant.
    Mais peut-on pousser le vice du polymorphisme paramétrique jusque là ?


    (Ou alors tout autre chose, ce que je peux peut-être faire, c'est forcer le type de retour. Tout dépend des cas d'utilisation. Je vais essayer de réfléchir à ça…)
    Cours : Initiation à CMake
    Projet : Scalpel, bibliothèque d'analyse de code source C++ (développement en cours)
    Ce message a été tapé avec un clavier en disposition bépo.

  13. #13
    Expert confirmé

    Inscrit en
    Novembre 2005
    Messages
    5 145
    Détails du profil
    Informations forums :
    Inscription : Novembre 2005
    Messages : 5 145
    Par défaut
    Citation Envoyé par Florian Goo Voir le message
    Finalement ça débouche peut-être sur une question plus simple : quand utiliser le pattern Visitor et quand utiliser boost::variant (ou autre) ?
    J'ai du mal à voir pourquoi tu opposes les deux.

    Le pattern visiteur, c'est quand tu as une structure de donnée et que tu veux appliquer une opération sur chaque élément contenu. On fait un object pour l'opération, la structure de donnée a un membre qui l'admet et l'applique à chaque élément -- ou à une sélection de ceux-ci. Si on a un arbre par exemple, on peut fournir des visiteurs pour une visite préfixe, infixe ou postfixe de l'arbre. Et un aspect variant dans les visiteurs, c'est commun ce contrôle -- choix et ordre des éléménts visités -- est fait; suivant les situations on le poussera plutôt dans la classe visitée ou dans le visiteur.

    Une situation très commune est que le type des éléments contenus peut varier dynamiquement. Comme l'opération dépend dynamiquement aussi du visiteur on a besoin d'un double dispatch. Le C++ n'en ayant pas, on présente souvent ce pattern avec un des idiomes permettant le double dispatch. Cet aspect est tellement commun qu'il y a de la confusion qui s'est installée entre le pattern visiteur et l'idiome du double dispatch.

    boost::variant, c'est un moyen simple de bâtir un type somme, avec une interface visiteur pour appliquer une opération sur l'unique élément contenu dedans (dans un cas dégénéré comme ici où il n'y a qu'un élémént, la différence entre double dispatch et visiteur est plus une différence d'intention que de méthode).

    J'ai donc du mal à opposer les deux. Je vois mal à quoi j'opposerais le pattern visiteur en C++.

    Boost variant peut s'opposer à la construction d'une hiérarchie fermée (ce dont dépendent l'idiome le plus fréquent de dispatch multiple). Quelques éléments:

    - boost variant est bien adapté au sémantique de valeur, les hiérarchies à celles d'entité
    - boost variant est bien adapté quand les types rassemblés sont indépendants, quand ils ne le sont pas les hiérarchies rendent l'aspect commun plus visible

    Dans le cas de hiérarchie fermée, il faut aussi considérer la possibilité d'avoir un tag explicite (un membre indiquant de quelle classe il s'agit). Un test ou un switch sur celui-ci peut être plus clair qu'un double dispatch.

    Dans ton cas, je pencherai plutôt vers la hiérarchie, justement à cause des deux points donnés: on a affaire à des entités ayant un comportement en partie commun. Je considérerais la possibilité d'avoir un tag. En passant, on est dans un cas où je vois de l'héritage multiple arriver naturellement: une classe est un scope tout comme elle est un type.

Discussions similaires

  1. polymorphisme paramétrique en C
    Par denispir dans le forum Débuter
    Réponses: 5
    Dernier message: 04/04/2012, 13h39
  2. Fonction membre constante, cas litigieux..
    Par Bakura dans le forum C++
    Réponses: 13
    Dernier message: 11/05/2009, 13h40
  3. [conception]polymorphisme statique ou dynamique ?
    Par vandamme dans le forum C++
    Réponses: 7
    Dernier message: 15/07/2007, 10h14

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