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 :

Le programme ne produit pas l'output indiqué dans le livre.


Sujet :

C++

Vue hybride

Message précédent Message précédent   Message suivant Message suivant
  1. #1
    Membre éclairé
    Avatar de Pierre8r
    Homme Profil pro
    Inscrit en
    Octobre 2004
    Messages
    518
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 70
    Localisation : France, Paris (Île de France)

    Informations forums :
    Inscription : Octobre 2004
    Messages : 518
    Par défaut Le programme ne produit pas l'output indiqué dans le livre.
    Bonjour,

    Le programme ne produit pas l'output indiqué dans le livre.

    Soit le code suivant :

    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
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    #include <iostream>
    using namespace std ;
    class point
    {
        int x, y ;
    public :
        point (int abs=0, int ord=0)     // constructeur "usuel"
        {
            x=abs ;
            y=ord ;
            cout << "++ Appel Const. usuel   " << this << " " << x << " " << y << "\n" ;
        }
        point (const point & p)          // constructeur de recopie
        {
            x=p.x ;
            y=p.y ;
            cout << "++ Appel Const. recopie " << this << " " << x << " " << y << "\n" ;
        }
        ~point ()
        {
            cout << "-- Appel Destr.         " << this << " " << x << " " << y << "\n" ;
        }
        point symetrique () ;
    } ;
     
    point point::symetrique ()
    {
        point res ;
        res.x = -x ;
        res.y = -y ;
        return res ;
    }
     
    int main()
    {
        point a(1,3), b ;
        cout << "** avant appel de symetrique\n" ;
        b = a.symetrique () ;
        cout << "** apres appel de symetrique\n" ;
    }
    Il est indiqué que le code doit produire un output proche de ça :

    ++ Appel Const. usuel 0x61ff10 1 3
    ++ Appel Const. usuel 0x61ff08 0 0
    ** avant appel de symetrique
    ++ Appel Const. usuel 0x61ff18 0 0
    ++ Appel Const. recopie
    -- Appel Destr. 0x61ff18 -1 -3
    -- Appel Destr.
    ** apres appel de symetrique
    -- Appel Destr. 0x61ff08 -1 -3
    -- Appel Destr. 0x61ff10 1 3
    Mais l'output est celui-ci :
    ++ Appel Const. usuel 0x61ff10 1 3
    ++ Appel Const. usuel 0x61ff08 0 0
    ** avant appel de symetrique
    ++ Appel Const. usuel 0x61ff18 0 0
    -- Appel Destr. 0x61ff18 -1 -3
    ** apres appel de symetrique
    -- Appel Destr. 0x61ff08 -1 -3
    -- Appel Destr. 0x61ff10 1 3

  2. #2
    Expert confirmé
    Homme Profil pro
    Ingénieur développement matériel électronique
    Inscrit en
    Décembre 2015
    Messages
    1 600
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 62
    Localisation : France, Bouches du Rhône (Provence Alpes Côte d'Azur)

    Informations professionnelles :
    Activité : Ingénieur développement matériel électronique
    Secteur : High Tech - Électronique et micro-électronique

    Informations forums :
    Inscription : Décembre 2015
    Messages : 1 600
    Par défaut
    Bonjour,

    Ce qui se déroule dans ce cas dépend à la fois du compilateur et de l'évolution de la norme.
    Ici, il peut y avoir 4 objets créés ou 3 ou 2 (maximum 3 depuis C++17). L'auteur de ton bouquin ne semble pas connaître le RVO.
    Plus grave, il ne semble pas parler du constructeur par déplacement qui pourrait être utilisé ici ; concept apparu en C++11 qui peut être important de connaître, on pourrait avoir :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    ++ Appel Const. usuel   0113FCF8 1 3            // seul a est construit pour le moment
    ** avant appel de symetrique
    ++ Appel Const. usuel   0113FD00 0 0            // res et b sont un seul objet
    ** apres appel de symetrique
    -- Appel Destr.         0113FD00 -1 -3          // res et b sont détruits
    -- Appel Destr.         0113FCF8 1 3

  3. #3
    Expert éminent
    Avatar de Médinoc
    Homme Profil pro
    Développeur informatique
    Inscrit en
    Septembre 2005
    Messages
    27 398
    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 398
    Par défaut
    Pour traduire en clair, suite à la Return Value Optimization (RVO), l'appel b = a.symetrique () ; a créé la valeur directement dans b, éliminant le besoin de passer par un objet temporaire et copier l'objet.
    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
    Modérateur

    Avatar de Bktero
    Homme Profil pro
    Ingénieur développement logiciels
    Inscrit en
    Juin 2009
    Messages
    4 496
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 38
    Localisation : France, Loire Atlantique (Pays de la Loire)

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

    Informations forums :
    Inscription : Juin 2009
    Messages : 4 496
    Billets dans le blog
    1

  5. #5
    Membre éclairé
    Avatar de Pierre8r
    Homme Profil pro
    Inscrit en
    Octobre 2004
    Messages
    518
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 70
    Localisation : France, Paris (Île de France)

    Informations forums :
    Inscription : Octobre 2004
    Messages : 518
    Par défaut
    Merci à tous.
    J'ai essayé Build options... avec C++98 dans Code::Block et j'ai eu le même résultat.

  6. #6
    Expert confirmé
    Homme Profil pro
    Ingénieur développement matériel électronique
    Inscrit en
    Décembre 2015
    Messages
    1 600
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 62
    Localisation : France, Bouches du Rhône (Provence Alpes Côte d'Azur)

    Informations professionnelles :
    Activité : Ingénieur développement matériel électronique
    Secteur : High Tech - Électronique et micro-électronique

    Informations forums :
    Inscription : Décembre 2015
    Messages : 1 600
    Par défaut
    L'option pour désactiver le RVO avec g++ est -fno-elide-constructors.

  7. #7
    Membre Expert
    Avatar de Pyramidev
    Homme Profil pro
    Tech Lead
    Inscrit en
    Avril 2016
    Messages
    1 515
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Haute Garonne (Midi Pyrénées)

    Informations professionnelles :
    Activité : Tech Lead

    Informations forums :
    Inscription : Avril 2016
    Messages : 1 515
    Par défaut
    Citation Envoyé par dalfab Voir le message
    Ici, il peut y avoir 4 objets créés ou 3 ou 2 (maximum 3 depuis C++17).
    Non, il y en a forcément au moins 3 appels au constructeur de point : un pour créer a, un pour créer b et un pour créer a.symetrique (). Dans l'instruction b = a.symetrique () ;, il ne peut pas y avoir d'optimisation spéciale pour modifier b sans appeler le constructeur de point.

    D'ailleurs, même en C++17, il peut y avoir 4 appels au constructeur : avec le code actuel de point::symetrique, le C++17 ne garantit pas l'élision de copie. Le C++17 garantit la RVO, mais pas la NRVO.

    Pour que le C++17 garantisse l'élision de copie, il faut que point::symetrique retourne une prvalue, par exemple comme ceci :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    point point::symetrique ()
    {
        return point(-x, -y);
    }
    Plus de détails : https://en.cppreference.com/w/cpp/language/copy_elision

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

Discussions similaires

  1. Réponses: 4
    Dernier message: 21/10/2017, 21h25
  2. Sos mon programme ne marche pas
    Par monalisa_dulouvre dans le forum C
    Réponses: 49
    Dernier message: 26/12/2006, 18h35
  3. Réponses: 4
    Dernier message: 09/09/2006, 21h48
  4. Quel langage pour programme ne nécessitant pas d'install ?
    Par burnedsoul dans le forum Langages de programmation
    Réponses: 5
    Dernier message: 09/03/2006, 19h23
  5. [Dev-Pascal] Ne produit pas d'exe
    Par portix dans le forum Autres IDE
    Réponses: 8
    Dernier message: 02/07/2003, 15h14

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