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 :

[template] passer un argument par valeur ou en const &


Sujet :

C++

Vue hybride

Message précédent Message précédent   Message suivant Message suivant
  1. #1
    Membre Expert
    Avatar de méphistopheles
    Profil pro
    Inscrit en
    Janvier 2005
    Messages
    1 551
    Détails du profil
    Informations personnelles :
    Âge : 38
    Localisation : France

    Informations forums :
    Inscription : Janvier 2005
    Messages : 1 551
    Par défaut [template] passer un argument par valeur ou en const &
    Bonjour.

    J'ai récemment écrit la classe de wrapper simple suivante:
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    template <class Wrapped>
    class Wrapper : public Wrapped
    {
      public:
      Wrapper():Wrapped(){}
      template <class T>
      Wrapper(T t):Wrapped(t)
      {
         char u;//juste pour déclencher un warning. 
      }
    };
    j'essaye ensuite d'exécuter ce code:
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
     
    int main()
    {
      std::string s;
      char f[32]="dummy"
      Wrapper<std::string> u(s);
      Wrapper<std::string> v("foo");
      Wrapper<std::string> w(f);
    }
    quand je regarde le message du compilo, j'ai:
    Citation Envoyé par gcc 4.4.1
    -------------- Build: Debug in Boost_Serialize_Test ---------------

    Compiling: main.cpp
    /.../main.cpp: In constructor ‘Wrapper<Wrapped>::Wrapper(T) [with T = std::basic_string<char, std::char_traits<char>, std::allocator<char> >, Wrapped = std::basic_string<char, std::char_traits<char>, std::allocator<char> >]’:
    /.../main.cpp:103: instantiated from here
    /.../main.cpp:39: warning: unused variable ‘u’
    /.../main.cpp: In constructor ‘Wrapper<Wrapped>::Wrapper(T) [with T = const char*, Wrapped = std::basic_string<char, std::char_traits<char>, std::allocator<char> >]’:
    /.../main.cpp:104: instantiated from here
    /.../main.cpp:39: warning: unused variable ‘u’
    /.../main.cpp: In constructor ‘Wrapper<Wrapped>::Wrapper(T) [with T = char*, Wrapped = std::basic_string<char, std::char_traits<char>, std::allocator<char> >]’:
    /.../main.cpp:105: instantiated from here
    /.../main.cpp:39: warning: unused variable ‘u’
    Linking console executable: bin/Debug/Boost_Serialize_Test
    Output size is 252,76 KB
    Process terminated with status 0 (0 minutes, 7 seconds)
    0 errors, 4 warnings
    La partie en rouge montre que le constructeur prend pour string un passage par valeur... or j'aimerais bien avoir, dans ce cas, un passage par référence de type const std::string &...

    il faudrait pour cela spécialiser le template, je suppose sur un critère de taille du parametre wrapped. Je me souviens avoir déjà vu ce genre de bout de code sur le forum, mais impossible de le retrouver ... si vous savez ou le trouver ou comment faire , merci d'avance

  2. #2
    Expert confirmé
    Homme Profil pro
    Développeur informatique
    Inscrit en
    Février 2005
    Messages
    5 468
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 53
    Localisation : France, Val de Marne (Île de France)

    Informations professionnelles :
    Activité : Développeur informatique
    Secteur : Conseil

    Informations forums :
    Inscription : Février 2005
    Messages : 5 468
    Par défaut
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    Wrapper(T t):Wrapped(t)


    Ne serait-ce pas plutôt ça :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    Wrapper(const T& t):Wrapped(t)

  3. #3
    Expert éminent
    Avatar de Médinoc
    Homme Profil pro
    Développeur informatique
    Inscrit en
    Septembre 2005
    Messages
    27 394
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 41
    Localisation : France

    Informations professionnelles :
    Activité : Développeur informatique
    Secteur : High Tech - Éditeur de logiciels

    Informations forums :
    Inscription : Septembre 2005
    Messages : 27 394
    Par défaut
    Si tu veux systématiquement un const T&, il suffit de demander.

    Sinon, il faut une classe de traits, un truc de ce genre:

    Code C++ : 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
    template< class T, bool IsBig >
    class Param2
    {
    public:
    	typedef T type;
    };
     
    //Note: je ne me souviens plus de la syntaxe pour les spécialisations partielles de classe template
    template< class T >
    class Param2< T, true >
    {
    public:
    	typedef T const &type;
    };
     
    template< class T >
    class Param
    {
    public:
    	typedef typename Param2< T, (sizeof(T)>sizeof(long long))>::type type;
    };
     
     
    template< class T >
    void UneFonctionTemplate(typename Param<T>::type toto)
    {
    	//Blabla
    }
    SVP, pas de questions techniques par MP. Surtout si je ne vous ai jamais parlé avant.

    "Aw, come on, who would be so stupid as to insert a cast to make an error go away without actually fixing the error?"
    Apparently everyone.
    -- Raymond Chen.
    Traduction obligatoire: "Oh, voyons, qui serait assez stupide pour mettre un cast pour faire disparaitre un message d'erreur sans vraiment corriger l'erreur?" - Apparemment, tout le monde. -- Raymond Chen.

  4. #4
    Membre Expert
    Avatar de Goten
    Profil pro
    Inscrit en
    Juillet 2008
    Messages
    1 580
    Détails du profil
    Informations personnelles :
    Âge : 34
    Localisation : France

    Informations forums :
    Inscription : Juillet 2008
    Messages : 1 580
    Par défaut
    ou boost::call_traits<T>::param_type ... .

    @Medinoc : c'est bien ça la syntaxe pour la spécialisation partielle.

  5. #5
    Membre Expert
    Avatar de méphistopheles
    Profil pro
    Inscrit en
    Janvier 2005
    Messages
    1 551
    Détails du profil
    Informations personnelles :
    Âge : 38
    Localisation : France

    Informations forums :
    Inscription : Janvier 2005
    Messages : 1 551
    Par défaut
    Citation Envoyé par Médinoc Voir le message
    Sinon, il faut une classe de traits, un truc de ce genre:
    c'est effectivement ce qu'il me fallait. la spécialisation est bonne et marche bien.... mais je ne peux pas utiliser le type: je ne comprend pas:
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    template< class T >
    class Param
    {
    public:
    	typedef typename Param2< T, (sizeof(T)>sizeof(long long))>::type type;
            //error: type ‘Param2<T, (sizeof (T) > 8u)>’ is not derived from type ‘Param<T>’|
            ///error: expected ‘;’ before ‘type’|
    };
    je vais voir pour boost ....

    Edit: quelqu'un pourrai-til tester chez lui pour voir si ça vient de moi ?
    merci

  6. #6
    Membre Expert
    Avatar de méphistopheles
    Profil pro
    Inscrit en
    Janvier 2005
    Messages
    1 551
    Détails du profil
    Informations personnelles :
    Âge : 38
    Localisation : France

    Informations forums :
    Inscription : Janvier 2005
    Messages : 1 551
    Par défaut
    je viens de tester avec boost j'ai exactement le même problème :
    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
    #include <boost/call_traits.hpp>
    #include <string>
    template <class Wrapped>
    class Wrapper : public Wrapped
    {
      public:
      Wrapper():Wrapped(){}
     
      template <class T>
      Wrapper( boost::call_traits<T>::param_type  t) //error: expected ‘)’ before ‘t’
        :Wrapped(t)
      {
        char u;
      }
    };
     
     
    int main()
    {
      Wrapper<std::string> * x=new Wrapper<std::string>("blabla");
      return 0;
    }
    avec comme erreur:
    Citation Envoyé par gcc
    -------------- Build: Debug in Boost_Serialize_Test ---------------

    Compiling: main.cpp
    /.../main.cpp:41: error: expected ‘)’ before ‘t’
    /.../main.cpp: In function ‘int main()’:
    /.../main.cpp:101: error: no matching function for call to ‘Wrapper<std::basic_string<char, std::char_traits<char>, std::allocator<char> > >::Wrapper(const char [24])’
    /.../main.cpp:38: note: candidates are: Wrapper<Wrapped>::Wrapper() [with Wrapped = std::basic_string<char, std::char_traits<char>, std::allocator<char> >]
    /.../main.cpp:36: note: Wrapper<std::basic_string<char, std::char_traits<char>, std::allocator<char> > >::Wrapper(const Wrapper<std::basic_string<char, std::char_traits<char>, std::allocator<char> > >&)
    /.../main.cpp:101: warning: unused variable ‘x’
    Process terminated with status 1 (0 minutes, 1 seconds)
    2 errors, 1 warnings
    (note: les numéros de ligne ne sont pas les bons, car j'ai retiré des commentaires).

    quelqu'un a une idée ?

    merci

  7. #7
    Expert confirmé
    Homme Profil pro
    Développeur informatique
    Inscrit en
    Février 2005
    Messages
    5 468
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 53
    Localisation : France, Val de Marne (Île de France)

    Informations professionnelles :
    Activité : Développeur informatique
    Secteur : Conseil

    Informations forums :
    Inscription : Février 2005
    Messages : 5 468
    Par défaut
    Truc très con, mais qui m'a souvent dépanné dans le passé, c'est d'utiliser l'option "/P" de VS pour avoir le résultat du pré-processing de l'unité de compilation.
    C'est permet de voir des remplacements via MACRO ou des ; manquant en fin de fichier d'include, etc...

    Donc quand je comprends rien, je fais cette astuce pour voir si c'est moi ou une connerie dans un fichier d'en-tête connu ni d'Eve ni des dents.

  8. #8
    Membre Expert
    Avatar de méphistopheles
    Profil pro
    Inscrit en
    Janvier 2005
    Messages
    1 551
    Détails du profil
    Informations personnelles :
    Âge : 38
    Localisation : France

    Informations forums :
    Inscription : Janvier 2005
    Messages : 1 551
    Par défaut
    heu...contrairement à visual gcc n'inclut rien de base normalement (a part les librairies iso, mais pas les headers), enfin je vais essayer


    Edit: c'est pas les options de compilation : même sur codepad, j'ai ça... personne n'a une idée sur la nature de l'erreur ?

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

Discussions similaires

  1. template, itérateurs et argument par défaut de fonction
    Par goran kajfes dans le forum Langage
    Réponses: 5
    Dernier message: 24/05/2009, 18h00
  2. Passer des arguments par les signaux des event-box
    Par black is beautiful dans le forum GTK+ avec C & C++
    Réponses: 2
    Dernier message: 30/10/2008, 08h43
  3. Passage d'arguments par valeur et adresse
    Par ToutEnMasm dans le forum C++
    Réponses: 12
    Dernier message: 04/10/2007, 09h32
  4. Passage des arguments par valeur
    Par mpereg dans le forum Général Python
    Réponses: 4
    Dernier message: 13/03/2007, 17h12
  5. Réponses: 8
    Dernier message: 30/05/2005, 13h55

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