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 :

spécialisation de template en fonction de l'héritage


Sujet :

Langage C++

Vue hybride

Message précédent Message précédent   Message suivant Message suivant
  1. #1
    yan
    yan est déconnecté
    Rédacteur
    Avatar de yan
    Homme Profil pro
    Ingénieur expert
    Inscrit en
    Mars 2004
    Messages
    10 035
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 43
    Localisation : France, Ille et Vilaine (Bretagne)

    Informations professionnelles :
    Activité : Ingénieur expert
    Secteur : High Tech - Multimédia et Internet

    Informations forums :
    Inscription : Mars 2004
    Messages : 10 035
    Par défaut spécialisation de template en fonction de l'héritage
    Bonjour.
    Je cherche à spécialiser une template en fonction que la classe hérite ou non d'une autre classe.

    J'ai trouvé pour l'héritage avec une classe Traits
    http://www.gotw.ca/publications/mxc++-item-4.htm

    Mais je n'arrive pas à trouver comment aller plus loin.

    En gros je voudrais :
    1-dans le cas générale
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    template<typename T>
    class toto
    {
    T * m_ptr;
     
    /*...*/
    }
    2- si T hérite de A
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    template<typename T>
    class toto
    {
    titi<T> m_ptr;
     
    /*...*/
    }

    Je pense qu'il me faut une autre classe traits pour m_ptr, mais je ne voie pas comment mettre cela en oeuvre avec le test d'héritage..

    Y as peut être une meilleur solution...


  2. #2
    yan
    yan est déconnecté
    Rédacteur
    Avatar de yan
    Homme Profil pro
    Ingénieur expert
    Inscrit en
    Mars 2004
    Messages
    10 035
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 43
    Localisation : France, Ille et Vilaine (Bretagne)

    Informations professionnelles :
    Activité : Ingénieur expert
    Secteur : High Tech - Multimédia et Internet

    Informations forums :
    Inscription : Mars 2004
    Messages : 10 035
    Par défaut
    j'ai trouvé, avec une autre template interne qui se spécialise en fonction du résultat de l'héritage.
    Ca donne

    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
    template<typename T>
    class toto
    {
     
       template <bool P1,typename P2>
        struct interptr
        {
        };
     
        template<typename P2>
        struct interptr<false,P2>
        {
            P2   * ptr; 
            P2* get() {return ptr;}
            const P2* get()const {return ptr;}
        };
        template<typename P2>
        struct interptr<true,P2>
        {
            QWeakPointer<typename P2> ptr; 
             P2* get() {return ptr.data();}
            const P2* get()const {return ptr.data();}
        };
     
        interptr<IsDerivedFromHelper<T,QObject>::Is , T> m_ptr; 
     
     
    ...
     
    };

  3. #3
    Membre éclairé
    Inscrit en
    Avril 2005
    Messages
    1 110
    Détails du profil
    Informations forums :
    Inscription : Avril 2005
    Messages : 1 110
    Par défaut
    Je tombe à la renverse en voyant ce 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
    // Example 4-3(c): An IsDerivedFrom constraints base 
    // with testable value
    //
    template<typename D, typename B>
    class IsDerivedFrom
    {
      class No { };
      class Yes { No no[3]; }; 
    
      static Yes Test( B* ); // not defined
      static No Test( ... ); // not defined 
    
      static void Constraints(D* p) { B* pb = p; pb = p; } 
    
    public:
      enum { Is = sizeof(Test(static_cast<D*>(0))) == sizeof(Yes) }; 
    
      IsDerivedFrom() { void(*p)(D*) = Constraints; }
    };
    Il est donc possible de demander au compilateur d'exécuter une des méthodes Test() afin d'en évaluer la valeur retour par un sizeof(). Sidérant !

  4. #4
    Rédacteur
    Avatar de Laurent Gomila
    Profil pro
    Développeur informatique
    Inscrit en
    Avril 2003
    Messages
    10 651
    Détails du profil
    Informations personnelles :
    Âge : 41
    Localisation : France, Moselle (Lorraine)

    Informations professionnelles :
    Activité : Développeur informatique

    Informations forums :
    Inscription : Avril 2003
    Messages : 10 651
    Par défaut
    Il est donc possible de demander au compilateur d'exécuter une des méthodes Test() afin d'en évaluer la valeur retour par un sizeof(). Sidérant !
    Non. Simplement de résoudre l'appel, trouver la bonne surcharge et donc le type de retour correspondant à cette surcharge. Rien n'est évalué, le compilo ne fait que son boulot habituel ici. D'ailleurs tu vois bien que les fonctions Test() n'ont aucune définition, il n'y a rien à appeler.

  5. #5
    yan
    yan est déconnecté
    Rédacteur
    Avatar de yan
    Homme Profil pro
    Ingénieur expert
    Inscrit en
    Mars 2004
    Messages
    10 035
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 43
    Localisation : France, Ille et Vilaine (Bretagne)

    Informations professionnelles :
    Activité : Ingénieur expert
    Secteur : High Tech - Multimédia et Internet

    Informations forums :
    Inscription : Mars 2004
    Messages : 10 035
    Par défaut
    Citation Envoyé par camboui Voir le message
    Il est donc possible de demander au compilateur d'exécuter une des méthodes Test() afin d'en évaluer la valeur retour par un sizeof(). Sidérant !
    Je crois que le sizeof vérifie juste le taille du type de retour.

    Moi ce qui me choque c'est que l'on as deux fonctions static Test qui ne retournent pas la même chose, ne prennent pas le même arguments et qui ne sont pas définie

    Y as boost qui propose un truc,mais y as tellement d'include que j'ai pas trouvé comment ils font

  6. #6
    Membre éclairé
    Inscrit en
    Avril 2005
    Messages
    1 110
    Détails du profil
    Informations forums :
    Inscription : Avril 2005
    Messages : 1 110
    Par défaut
    Je sais bien que rien n'est évalué, itaque j'ai mis le verbe en italique.

    N'empêche, vous semblez blasé de voir ce genre d'écriturealors que foo() serait juste déclarée (et même pas définie comme le souligne yan).
    C'est la première fois que je vois autre chose qu'une variable ou un type explicite dans un sizeof().
    On peut donc y ajouter le type retour d'une fonction. Y'en a d'autres comme ça ?

  7. #7
    Rédacteur
    Avatar de Laurent Gomila
    Profil pro
    Développeur informatique
    Inscrit en
    Avril 2003
    Messages
    10 651
    Détails du profil
    Informations personnelles :
    Âge : 41
    Localisation : France, Moselle (Lorraine)

    Informations professionnelles :
    Activité : Développeur informatique

    Informations forums :
    Inscription : Avril 2003
    Messages : 10 651
    Par défaut
    Moi ce qui me choque c'est que l'on as deux fonctions static Test qui ne retournent pas la même chose, ne prennent pas le même arguments et qui ne sont pas définie
    - "retournent pas la même chose, ne prennent pas le même arguments" --> ce sont deux surcharges, quoi
    - "ne sont pas définie" --> ça poserait un problème à l'éditeur de liens lorsqu'il rencontrerait un appel, mais comme le compilateur n'en a pas généré, il n'y a pas besoin de définition

    Y as boost qui propose un truc,mais y as tellement d'include que j'ai pas trouvé comment ils font
    La plupart des détections utilisant SFINAE s'implémentent très exactement avec ce type de code, y compris dans boost

  8. #8
    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
    Par ailleurs yan, tu trouveras pas mal de templates pour savoir si une classe dérive d'une autre etc dans boost.type_traits.

  9. #9
    yan
    yan est déconnecté
    Rédacteur
    Avatar de yan
    Homme Profil pro
    Ingénieur expert
    Inscrit en
    Mars 2004
    Messages
    10 035
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 43
    Localisation : France, Ille et Vilaine (Bretagne)

    Informations professionnelles :
    Activité : Ingénieur expert
    Secteur : High Tech - Multimédia et Internet

    Informations forums :
    Inscription : Mars 2004
    Messages : 10 035
    Par défaut
    Citation Envoyé par Laurent Gomila Voir le message
    - "retournent pas la même chose, ne prennent pas le même arguments" --> ce sont deux surcharges, quoi
    Il me semblais que le type retourné ne pouvais pas être différent

    SFINAE.... va bien falloire que je comprenne ce que c'est réellement un de ces jours

  10. #10
    Rédacteur
    Avatar de 3DArchi
    Profil pro
    Inscrit en
    Juin 2008
    Messages
    7 634
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Juin 2008
    Messages : 7 634
    Par défaut
    Salut,
    Citation Envoyé par Laurent Gomila Voir le message
    - "ne sont pas définie" --> ça poserait un problème à l'éditeur de liens lorsqu'il rencontrerait un appel, mais comme le compilateur n'en a pas généré, il n'y a pas besoin de définition
    Et le fait que ce ne soit pas défini indique bien que tout est fait à la compilation et qu'aucun code supplémentaire n'est ajouté à l'exécutable cible. Donc, coût metaprog == 0

  11. #11
    Membre chevronné Avatar de Lavock
    Profil pro
    Inscrit en
    Octobre 2009
    Messages
    560
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Octobre 2009
    Messages : 560
    Par défaut
    C'est peut-être stupide, mais à quoi serve ces lignes :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
     
      static void Constraints(D* p) { B* pb = p; pb = p; } 
    };
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
     
      IsDerivedFrom() { void(*p)(D*) = Constraints; }
    };
    ?!?(surtout la première qui fait la même instruction 2 fois ><) ?

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

Discussions similaires

  1. Spécialisation partielle d'une fonction template
    Par N0vember dans le forum Langage
    Réponses: 4
    Dernier message: 17/10/2010, 20h49
  2. Réponses: 8
    Dernier message: 18/06/2007, 15h06
  3. Réponses: 8
    Dernier message: 01/03/2007, 15h07
  4. [POO] Pointeur sur fonction membre et héritage
    Par MrDuChnok dans le forum C++
    Réponses: 9
    Dernier message: 20/07/2006, 17h19
  5. Réponses: 7
    Dernier message: 01/01/2006, 03h28

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