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 :

pbe: specialisation de template/bibliotheque statique


Sujet :

C++

  1. #1
    Membre actif
    Profil pro
    Inscrit en
    Mai 2009
    Messages
    45
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Mai 2009
    Messages : 45
    Par défaut pbe: specialisation de template/bibliotheque statique
    Bonjour à tous,

    Je rencontre une erreur de link sur un point délicat de C++ sur un projet conséquent, et j'ai du mal à comprendre cette erreur.

    J'ai réécrit un ECM qui reproduit le problème, constitué de 3 fichiers:
    un fichier contenant le main(), et d'une bibliothèque statique mylib.a, construite à partir de mylib.h et mylib.cpp

    le main:
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    #include "mylib.hpp"
    int main()
    {
        std::cout << "hello world!\n";
        foo<int>();
        foo<float>();
        SomeFunc();
    }
    Le fichier d'en tete:
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    #ifndef MYLIB_HPP
    #define MYLIB_HPP
     
    #include <iostream>
     
    template <typename T>
    void foo()
    {
        std::cout << "generic foo\n";
    }
    void SomeFunc();
    #endif
    Le code de mylib.cpp associé
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    #include "mylib.hpp"
     
    void SomeFunc()
    {
        std::cout << "SomeFunc\n";
    }
    Jusque là, tout va bien... Je peux générer la bibliothèque statique, et compiler et exécuter le main().

    Le problème apparaît quand j'essaie de spécialiser 'foo()': si j'ajoute dans le .h une spécialisation:

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    template <>
    void foo<int>()
    {
        std::cout << " int foo\n";
    }
    Alors le linker me jette au motif d'une double définition de 'foo()', dans le main.cpp et dans mylib.cpp. Or, j'ai bien les "include-guards" !
    Et pourquoi ce problème apparait-il uniquement en cas de spécialisation ? Cela voudrait dire que la fonction générique n'est pas instanciée dans mylib.a, mais que la spécialisation l'est!
    Je ne comprend pas pourquoi, alors que la fonction n'est pas instanciée (?) dans mylib.a

    Merci pour votre sagacité à résoudre ce mystère...

  2. #2
    Membre expérimenté Avatar de Rewpparo
    Homme Profil pro
    Amateur
    Inscrit en
    Décembre 2005
    Messages
    170
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 44
    Localisation : France, Charente Maritime (Poitou Charente)

    Informations professionnelles :
    Activité : Amateur

    Informations forums :
    Inscription : Décembre 2005
    Messages : 170
    Par défaut
    Les spécialisations totales de template vont dans le fichier .cpp, soit dans la lib soit dans le programme.
    Si tu le mets dans le .h, que tu inclus le .h dans un cpp de la lib, il le compile dedans. Quand tu inclus le .h dans le main, il le recompile une autre fois. Au linkage, il a deux instances.
    En fait une spécialisation de template se comporte de ce point de vue comme une définition classique de fonction.

  3. #3
    Membre actif
    Profil pro
    Inscrit en
    Mai 2009
    Messages
    45
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Mai 2009
    Messages : 45
    Par défaut
    Citation Envoyé par Rewpparo Voir le message
    En fait une spécialisation de template se comporte de ce point de vue comme une définition classique de fonction.
    Bon, effectivement, si je mets la spécialisation dans mylib.cpp, l'erreur disparait.

    Ce qui m'embete, c'est que du coup la définition des fonctions va être étalée dans plusieurs fichiers, les versions génériques dans le .h et les spécialisées dans le .cpp. Pas trop le choix, j'imagine...

    Merci, en tout cas, ca faisait quelques jours que je bloquais là-dessus!

  4. #4
    Membre expérimenté Avatar de Rewpparo
    Homme Profil pro
    Amateur
    Inscrit en
    Décembre 2005
    Messages
    170
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 44
    Localisation : France, Charente Maritime (Poitou Charente)

    Informations professionnelles :
    Activité : Amateur

    Informations forums :
    Inscription : Décembre 2005
    Messages : 170
    Par défaut
    Pour plus de clarté, il me semble que tu peux mettre le prototype de la spécialisation dans le .h. A vérifier.

  5. #5
    Membre actif
    Profil pro
    Inscrit en
    Mai 2009
    Messages
    45
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Mai 2009
    Messages : 45
    Par défaut
    Citation Envoyé par Rewpparo Voir le message
    Pour plus de clarté, il me semble que tu peux mettre le prototype de la spécialisation dans le .h. A vérifier.
    Après vérification, c'est même indispensable !

    J'avais cliqué sur "résolu" un peu vite, en fait l'erreur de build disparaissait, mais le comportement était erroné: un appel a foo<int>() executait la version générique au lieu de la spécialisée !

  6. #6
    Membre expérimenté Avatar de Rewpparo
    Homme Profil pro
    Amateur
    Inscrit en
    Décembre 2005
    Messages
    170
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 44
    Localisation : France, Charente Maritime (Poitou Charente)

    Informations professionnelles :
    Activité : Amateur

    Informations forums :
    Inscription : Décembre 2005
    Messages : 170
    Par défaut
    Hum normalement non, il me semble que le linker résout ces liens la a la fin, donc le prototype il s'en tamponne.
    A vérifier par les gourous qui hantent ces lieux.

  7. #7
    Membre Expert
    Avatar de Klaim
    Homme Profil pro
    Développeur de jeux vidéo
    Inscrit en
    Août 2004
    Messages
    1 717
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France

    Informations professionnelles :
    Activité : Développeur de jeux vidéo
    Secteur : High Tech - Éditeur de logiciels

    Informations forums :
    Inscription : Août 2004
    Messages : 1 717
    Par défaut
    Une specialisation est comme une fonction normale, elle suit donc les memes regles :

    - soit tu la declares dans le h et tu la definis dans le cpp;
    - soit tu la mets dans le h et tu la marque inline;

    la seconde est la plus simple, la premiere peut etre pas mal si la fonction fait beaucoup de choses.

  8. #8
    Rédacteur/Modérateur
    Avatar de JolyLoic
    Homme Profil pro
    Développeur informatique
    Inscrit en
    Août 2004
    Messages
    5 463
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 51
    Localisation : France, Yvelines (Île de France)

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

    Informations forums :
    Inscription : Août 2004
    Messages : 5 463
    Par défaut
    Citation Envoyé par Rewpparo Voir le message
    Hum normalement non, il me semble que le linker résout ces liens la a la fin, donc le prototype il s'en tamponne.
    A vérifier par les gourous qui hantent ces lieux.
    Le choix de la fonction à appeler est effectué à la phase de compilation, rien qu'en lisant les .h. A la phase de lien, le linker a juste pour tâche d'associer une adresse à la fonction qui a déjà été choisie au préalable (c'est plus compliqué, puisqu'il peut aussi faire de l'inlining, ou d'autres optimisations).
    Ma session aux Microsoft TechDays 2013 : Développer en natif avec C++11.
    Celle des Microsoft TechDays 2014 : Bonnes pratiques pour apprivoiser le C++11 avec Visual C++
    Et celle des Microsoft TechDays 2015 : Visual C++ 2015 : voyage à la découverte d'un nouveau monde
    Je donne des formations au C++ en entreprise, n'hésitez pas à me contacter.

  9. #9
    Membre expérimenté Avatar de Rewpparo
    Homme Profil pro
    Amateur
    Inscrit en
    Décembre 2005
    Messages
    170
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 44
    Localisation : France, Charente Maritime (Poitou Charente)

    Informations professionnelles :
    Activité : Amateur

    Informations forums :
    Inscription : Décembre 2005
    Messages : 170
    Par défaut
    C'est bizarre, dans mon code actuel j'ai quelque chose comme ca :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    ///Window.h
    class Window
    {
    public:
       template <unsigned RENDERSYSTEM> Window* create();
    };
     
    ///WindowSDL.cpp
    template <> Window* Window::create<RENDERSYSTEM_OPENGL>()
    {
       return new WindowSDL;
    }
    Je n'ai pas décri la spécialisation dans le .h, c'est d'ailleurs le but. Pourtant il me créé bien une fenêtre SDL. C'est parce que je n'ai pas d'implémentation générique ? Ou est-ce parce que la définition de la fonction template de l'OP est inline ?

  10. #10
    Rédacteur/Modérateur
    Avatar de JolyLoic
    Homme Profil pro
    Développeur informatique
    Inscrit en
    Août 2004
    Messages
    5 463
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 51
    Localisation : France, Yvelines (Île de France)

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

    Informations forums :
    Inscription : Août 2004
    Messages : 5 463
    Par défaut
    Tu appelles cette fonction depuis l'extérieur de WindowSDL.cpp ?
    Ma session aux Microsoft TechDays 2013 : Développer en natif avec C++11.
    Celle des Microsoft TechDays 2014 : Bonnes pratiques pour apprivoiser le C++11 avec Visual C++
    Et celle des Microsoft TechDays 2015 : Visual C++ 2015 : voyage à la découverte d'un nouveau monde
    Je donne des formations au C++ en entreprise, n'hésitez pas à me contacter.

  11. #11
    Membre expérimenté Avatar de Rewpparo
    Homme Profil pro
    Amateur
    Inscrit en
    Décembre 2005
    Messages
    170
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 44
    Localisation : France, Charente Maritime (Poitou Charente)

    Informations professionnelles :
    Activité : Amateur

    Informations forums :
    Inscription : Décembre 2005
    Messages : 170
    Par défaut
    WindowSDL.cpp est dans une lib statique, et je l'appelle depuis le programme principal. Testé g++ et MSVC 2010.

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

Discussions similaires

  1. Bibliotheque statique à partir d'une dynamique
    Par hebus44 dans le forum Autres éditeurs
    Réponses: 5
    Dernier message: 26/05/2009, 00h18
  2. specialisation de template
    Par camboui dans le forum Langage
    Réponses: 2
    Dernier message: 29/01/2009, 14h16
  3. compilation bibliotheque dynamique contenant une bibliotheque statique
    Par L'elfe d'Azur dans le forum Autres éditeurs
    Réponses: 0
    Dernier message: 28/10/2008, 11h00
  4. C++ dans C, probleme avec la bibliotheque statique
    Par L'elfe d'Azur dans le forum C
    Réponses: 2
    Dernier message: 26/06/2008, 10h10
  5. Réponses: 2
    Dernier message: 27/07/2007, 10h08

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