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 :

Mapping type - object


Sujet :

C++

Vue hybride

Message précédent Message précédent   Message suivant Message suivant
  1. #1
    Membre chevronné
    Homme Profil pro
    Ingénieur développement logiciels
    Inscrit en
    Mai 2006
    Messages
    366
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 42
    Localisation : France, Paris (Île de France)

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

    Informations forums :
    Inscription : Mai 2006
    Messages : 366
    Par défaut Mapping type - object
    Bonsoir,

    En voulant implémenter les algorithmes de la bibliothèques standards afin de pouvoir les utiliser avec des fonctions prenant un ou plusieurs arguments, je me suis naturellement orientés vers boost/bind. Un bout de code étant plus parlant qu'une longue explication, voici ce que j'obtiens pour l'algorithe for_each, acceptant un pointeur de membre prenant lui même un argument :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
     
    template<Iterator It, class T, class R, class Op>
    for_each(It first, It last, R (T::*pmf), boost::call_traits<Op>::parameter_type op) {
        std::for_each(first, last, boost::bind(pmf, _1, op));
    }
    Le problème est que boost::bind contient une copie interne des paramètres qu'on lui passe (ici op). DU coup l'avantage de l'utilisation de call_traits est perdu. Ma question est de savoir comment invoquer
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
     
    std::for_each(first, last, boost::bind(pmf, _1, cref(op)))
    plutôt que le code précédent lorsque call_traits<Op>::parameter_type est une référence. La spécialisation de template n'est pas vraiment une option envisageable, étant donné qu'avec un pointeur de fonction membre acceptant beaucoup de paramètres, on va devoir écrire beaucoup de code. Idem pour des structures if testant call_traits<Op>::parameter_type. Y'a-t-il une solution plus élégante ?

    Merci d'avance.

  2. #2
    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
    En voulant implémenter les algorithmes de la bibliothèques standards afin de pouvoir les utiliser avec des fonctions prenant un ou plusieurs arguments
    J'ai du mal à saisir l'intérêt ?

    Ma question est de savoir comment invoquer
    [...]
    plutôt que le code précédent lorsque call_traits<Op>::parameter_type est une référence
    Au pif de tête au réveil :

    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
    template <typename T> struct Truc
    {
        static T Param(T op) {return op;}
    };
    template <typename T> struct Truc<T&>
    {
        static ... Param(T& op) {return cref(op);}
    };
    template <typename T> struct Truc<const T&>
    {
        static ... Param(const T& op) {return cref(op);}
    };
     
    template<Iterator It, class T, class R, class Op>
    for_each(It first, It last, R (T::*pmf), boost::call_traits<Op>::parameter_type op)
    {
        std::for_each(first, last, boost::bind(pmf, _1, Truc<...>::Param(op)));
    }

  3. #3
    Membre chevronné
    Homme Profil pro
    Ingénieur développement logiciels
    Inscrit en
    Mai 2006
    Messages
    366
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 42
    Localisation : France, Paris (Île de France)

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

    Informations forums :
    Inscription : Mai 2006
    Messages : 366
    Par défaut
    L'interet est de :
    1) Restreindre la dependance a bind dans un seul fichier
    2) Alleger l'ecriture du code :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
     
    util::for_each(list.begin(), list.end(), &T::pmf, arg);
    est plus agreable que

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
     
    std::for_each(list.begin(), list.end(), boost::bind(&T::pmf, _1, arg));
    (enfin je trouve :p)

    Sinon j'avais pense a une solution du meme genre que la tienne :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
     
    template <class T> struct bind_param_type {
     
    typedef typename boost::mpl::if_<boost::is_reference<T>,
                                     boost::reference_wrapper<T>,
                                     T>::type;
    }
     
    // ....
     
    std::for_each(first, last, bind_param_type<...>::type(arg);
    Mais la tienne permet en plus de differencier les references constantes des references non constantes (bon ce serait possible aussi avec la solution exposee mais ca deviendrait tout de suite beaucoup moins lisible), et est mois lourde a implementer :p. La seule chose qui m'ennuie est pour les parametres passes par valeur (dans les deux cas), on a une copie supplementaire, et je ne vois pas de moyen de l'eviter (je ne suis meme pas sur que ce soit possible). Donc a defaut de trouver comment eviter cette copie je vais utiliser ton implementation,

    Merci beaucoup pour l'aide.

  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
    A partir du moment où tu utilises boost::call_traits<T>::param_type, dont le but est de passer par valeur les paramètres pour lesquels ça ne coute "pas cher", alors faire une copie ou (en fait trois ici -- sans optimisation) ne devrait pas vraiment être un problème.

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

Discussions similaires

  1. conversion de type Object -> String
    Par wiss20000 dans le forum Langage
    Réponses: 3
    Dernier message: 11/04/2007, 00h15
  2. mapping type sql oracle
    Par willoi dans le forum Hibernate
    Réponses: 13
    Dernier message: 16/02/2007, 14h28
  3. Réponses: 5
    Dernier message: 05/02/2007, 15h04
  4. [C#] Conversion implicite de type object vers int
    Par alexking2005 dans le forum C#
    Réponses: 5
    Dernier message: 02/01/2007, 10h02
  5. [C++ .Net] Comment récupérer les données du type Object ?
    Par psau dans le forum Framework .NET
    Réponses: 3
    Dernier message: 01/08/2006, 15h59

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