M2103-TP9-EXO-0-Corrigé

/**
 *
 * @file   complex.h
 *
 * @author M. Laporte
 *
 * @date   30/04/2019
 *
 * @brief  declartion de la classe Complex
 *
 **/

#ifndef COMPLEX_H
#define COMPLEX_H

#include <stdexcept>
#include <cmath>
#include <iostream>

namespace nsMath
{
    const double pi = std::acos (-1);

    class Complex
    {
        double myA;
        double myB;
        double myRho;
        double myTheta;
      public :
        Complex (double a = 0.0, double b = 0.0, bool isPolaire = false);

        Complex operator *  (const Complex &amp; c) const noexcept;
        Complex operator /  (const Complex &amp; c) const throw (std::runtime_error);
        Complex operator +  (const Complex &amp; c) const noexcept;
        Complex operator -  (const Complex &amp; c) const noexcept;

        bool    operator == (const Complex &amp; c) const noexcept;
		
		void display (std::ostream &amp; os) const;

    }; // Complex

} // nsMath
#endif // COMPLEX_H

/**
 *
 * @file   complex.cpp
 
 *
 * @author M. Laporte
 *
 * @date   30/04/2019
 *
 * @brief  definitions de la classe Complex
 *
 **/

#define COMPLEX nsMath::Complex

#include <stdexcept>
#include <cmath>

#include "complex.h"

using namespace std;

COMPLEX::Complex (double a /* = 0.0*/, double b /* = 0.0*/, 
                  bool isPolaire /*= false*/)
/*    * /
try
{ 
    {
        if (!isPolaire)
        {
            myA     = a;
            myB     = b;
            myRho   = sqrt (myA * myA + myB * myB);
            if (! myRho) throw runtime_error ("création impossible");
            myTheta = myA >= 0 ? asin (myB / myRho) : pi - asin (myB / myRho);
        }
        else
        {
            myRho   = a;
            myTheta = b;
            myA     = myRho * cos (myTheta);
            myB     = myRho * sin (myTheta);
         }
    } // Complex()
}

/*     */
{
    if (!isPolaire)
    {
        myA     = a;
        myB     = b;
        myRho   = sqrt (myA * myA + myB * myB);
        myTheta = myA >= 0 ? asin (myB / myRho) : pi - asin (myB / myRho);
    }
    else
    {
        myRho   = a;
        myTheta = b;
        myA     = myRho * cos (myTheta);
        myB     = myRho * sin (myTheta);
     }
} // Complex()

COMPLEX COMPLEX::operator *  (const Complex &amp; c) const noexcept
{
    return Complex  (myA * c.myA - myB * c.myB, myA * c.myB + myB * c.myA);

} // operator *()

COMPLEX COMPLEX::operator /  (const Complex &amp; c) const throw (std::runtime_error)
{
    double denom = c.myA * c.myA - c.myB*c.myB + 2 * c.myA * c.myB;
    if (denom == 0.0) throw runtime_error ("division par 0");
    return Complex ((myA * c.myA - myB * c.myB) / denom,
                    (myA * c.myB + myB * c.myA) / denom);
}

COMPLEX COMPLEX::operator +  (const Complex &amp; c) const noexcept
{
    return Complex  (myA + c.myA, myB + c.myB);

} // operator +()

COMPLEX COMPLEX::operator -  (const Complex &amp; c) const noexcept
{
    return Complex  (myA - c.myA, myB - c.myB);

} // operator *()


bool COMPLEX::operator == (const Complex &amp; c) const noexcept
{
    return myA == c.myA &amp;&amp; myB == c.myB;

} // operator ==()

void COMPLEX::display (ostream &amp; os) const
{
	os << myA << " + i * " << myB;
	
} // display()

#undef COMPLEX

/**
 *
 * @file   main.cpp
 
 *
 * @author M. Laporte
 *
 * @date   30/04/2019
 *
 * @brief  definitions de la classe Complex
 *
 **/
#include <iostream>
#include "complex.h"

using namespace std;
using namespace nsMath;

int main()
{
    Complex c1 (1, 2);
	cout << "c1 = ";
	c1.display (cout);
    cout << endl;
    Complex c2 (2, 3);
	cout << "c2 = ";
	c2.display (cout);
	cout << endl;
	cout << "c1 * c2 = ";
	(c1 * c2).display (cout);
	cout << endl;
	cout << "c1 + c2 = ";
	(c1 + c2).display (cout);
	cout << endl;
	cout << "c1 - c2 = ";
	(c1 - c2).display (cout);
	cout << endl;
	cout << "c1 / c2 = ";
	(c1 / c2).display (cout);cout << endl;
    Complex c3 (0, 0);
	cout << "c3 = ";
	c3.display (cout);
	cout << endl;
	try
	{
	    cout << "c1 / c3 = ";
		(c1 / c3).display (cout);
		cout << endl;
	}
	catch (const runtime_error &amp; e)
	{
		cerr << "division par un Compex nul : " << e.what () << endl;
	}
		
    return 0;
}

M2103-TP9-EXO-1

Créer le projet ReseauSocial.

Depuis C++11, C++ offre une bibliothèque ctime, qui permet de grossièrement gérer l’heure et la date. Cette bibliothèque reprend les structures du C, dans l’espace de nom std. Pour obtenir la date du jour, il faut procéder en 2 étapes. D’abord, récupérer le nombre de secondes écoulées depuis le 01/01/1970, à l’aide de la fonction time() de profil (simplifié)

time (time_t * t);

qui s’utilise comme suit :

time_t t; 
time (&t);

Ensuite,on transforme le time_t en une structure struct tm (rappel: struct classe où tout est publique par défaut) à l’aide de la fonction localtime() de profil

tm * localtime (time_t * t);

qui transforme t, le nombre de secondes écoulées depuis le 01/01/1970, en une tm et s’utilise comme suit :

tm local = * localtime (& t);

nous allons encapsuler cette struct tm dans la classe Date, comme sur le schéma UML suivant, de manière à interdire, à l’utilisateur qui utiliserait cette classe Date, l’accès aux attributs de la struct tm :

On ajoutera à cette classe Date un constructeur par défaut qui construit une Date avec la date du jour.

Travail demandé

Ecrire complètement les fichiers Date.h et Date.cpp qui contiennent respectivement les déclarations et les définitions de la classe Date correspondant au schéma et aux explications, dans l’espace de noms std.
Tester en ajoutant la fonction suivante à l’espace de noms anonyme de votre main.cpp :

    void testDate (void)
    {
        Date date;
        cout << "Aujourd'hui nous sommes le : " 
             << date.getDay () << ' ' << date.getMonth () << ' ' << date.getYear ()
             << " et il est " << date.getHour () << "h " << date.getMinute () 
             << "mn et " <

Petit problème, le mois est le mois dernier et l'année est n'importe quoi. Voici ce que dit le man sur les attributs mis en cause dans le struct tm :

           struct tm {
               int tm_sec;    /* Seconds (0-60) */
               int tm_min;    /* Minutes (0-59) */
               int tm_hour;   /* Hours (0-23) */
               int tm_mday;   /* Day of the month (1-31) */
               int tm_mon;    /* Month (0-11) */
               int tm_year;   /* Year - 1900 */
               int tm_wday;   /* Day of the week (0-6, Sunday = 0) */
               int tm_yday;   /* Day in the year (0-365, 1 Jan = 0) */
               int tm_isdst;  /* Daylight saving time * /
           };

Corriger dans les 2 accesseurs concernés et tester à nouveau.

M2103-TP9-EXO-1-Corrigé

Fichier Date.h

/**
 *
 * \file     Date.h
 *
 * \authors  M. Laporte
 *
 * \date     24/05/2018
 *
 * \version  V1.0
 *
 * \brief    Declaration de la classe Date
 *
 **/
#ifndef __DATE_H__
#define __DATE_H__

#include   // struct tm

namespace std
{
    class Date : public tm
    {
      public :
        Date (void);

        unsigned getYear   (void) const;
        unsigned getMonth  (void) const;
        unsigned getDay    (void) const;
        unsigned getHour   (void) const;
        unsigned getMinute (void) const;
        unsigned getSecond (void) const;

    }; // Date
} // std

#endif /* __DATE_H__ */

Fichier Date.cpp

/**
 *
 * \file     Date.cpp
 *
 * \authors  M. Laporte
 *
 * \date     24/05/2018
 *
 * \version  V1.0
 *
 * \brief    Definitions de la classe Date
 *
 **/

#include "date.h"
std::Date::Date (void)
{
    time_t whatTime;
    time (& whatTime);
    tm local;
    local = * localtime (&whatTime);
    tm_year = local.tm_year; 
    tm_mon  = local.tm_mon;
    tm_mday = local.tm_mday;
    tm_hour = local.tm_hour;
    tm_min  = local.tm_min;
    tm_sec  = local.tm_sec;

    //*this = * ((Date *) localtime (&whatTime));

    tm_year += 1900;
    tm_mon += 1;

}

unsigned std::Date::getYear   (void) const { return tm_year; }
unsigned std::Date::getMonth  (void) const { return tm_mon;  }
unsigned std::Date::getDay    (void) const { return tm_mday; }
unsigned std::Date::getHour   (void) const { return tm_hour; }
unsigned std::Date::getMinute (void) const { return tm_min;  }
unsigned std::Date::getSecond (void) const { return tm_sec;  }


M2103-TP9-EXO-2

Le but de ce qui suit est de traduire en C++ le schéma UML suivant :

Dans l’espace de noms anonyme du fichier main.cpp déclarer la classe Message, comme elle est décrite sur le schéma UML. Y ajouter un constructeur qui se contente de remplir myContent avec un paramètre correspondant, et myDate, avec la date courante (constructeur par défaut de Date). La fonction displayContent() affiche à l’écran le contenu du message. Vous intégrerez directement les définitions des fonctions dans la classe.
Dans l’espace de noms anonyme, insérer les différentes classes (exercices suivants), de haut en bas, dans l’ordre des exercices. Normalement, seules des déclarations de classes vides seront à ajouter. Lorsque cela sera nécessaire un embryon de classe sera fourni.

Ajouter à la classe l’accesseur à l’attribut myDate.

Dans l’espace de noms anomyme ajouter les fonctions displayDate() suivante :

    void displayDate (const Date & date)
    {
        cout << date.getDay () << ' ' << date.getMonth () << ' ' << date.getYear ();
    
    } // displayDate()

et displayHour() suivante :

    void displayHour (const Date & date)
    {
        cout << date.getHour () << "h " << date.getMinute () 
             << "mn et " <

Tester avec la fonction testMessage() suivante, également à incorporer à l'espace de noms anonyme du fichier main.cpp :

    void testMessage (void)
    {
        Message message ("salut");
        message.displayContent ();
        cout << " posté le : ";
        displayDate (message.getDate ());
        cout << " à ";
        displayHour (message.getDate ());
        cout << endl;

    } // testMessage()}

M2103-TP9-EXO-2-Corrigé

    class Message
    {
        string myContent;
        Date   myDate;
      public :
        Message (const string & content)
            : myContent (content) {}
        void displayContent (void) const { cout << myContent; }

        Date getDate (void) const { return myDate; }

    }; // Message

M2103-TP9-EXO-3

Dans l’espace de noms anonyme du fichier main.cpp déclarer la classe PublicMessage, comme elle est décrite sur le schéma UML.
La fonction add() se contente d’incrémenter de 1 myLikers (nombre de personnes qui aiment ce message).
Ajouter l’accesseur à myNbLikers.
Tester avec la fonction testPublicMessage() suivante, à incorporer également à l’espace de noms anonyme.

    void testPublicMessage (void)
    {
        PublicMessage message ("salut");
        message.displayContent ();
        cout << " posté le : ";
        displayDate (message.getDate ());
        cout << " à ";
        displayHour (message.getDate ());
        cout << endl;
        for (unsigned i(0); i < 5; ++i)
            message.add ();
        cout << "Nb de likers : " << message.getNbLikers (); << endl;

    } // testPublicMessage()

M2103-TP9-EXO-3-Corrigé

    class PublicMessage : public Message
    {
        unsigned myNbLikers;
      public :
        PublicMessage (const string & content)
           : Message (content) /*, myNbLikers (0)*/ {}

        unsigned getNbLikers (void) const { return myNbLikers; }

        void add (void) { ++ myNbLikers; }

    }; // PublicMessage

M2103-TP9-EXO-4

Dans l’espace de noms anonyme du fichier main.cpp déclarer la classe PrivateMessage, comme elle est décrite sur le schéma UML. Le constructeur a la tâche supplémentaire d’initialiser le lien avec l’User.
Ajouter l’accesseur à ce lien.

Tester avec la fonction testPrivateMessage() et la classe User provisoire suivantes, à incorporer également à l’espace de noms anonyme.

    class User
    {
       public :
       void sendMessage (const PrivateMessage & message)
       {
           message.displayContent ();
           cout << endl;
       } // sendMessage()
       void addFriend (User * oneFriend) { cout << "ajouté" << endl; }
       User * getFriend (unsigned i) const { cout << i << endl; return new User; }
       System * getSystem (void) const { cout << "système" << endl; return NULL; }
       void addMessage (const PrivateMessage & message) { message.displayContent (); cout << endl; }
       
    }; // User
    
    void testPrivateMessage (void)
    {
        User user;
        PrivateMessage message ("salut", & user);
        message.displayContent ();
        cout << " posté le : ";
        displayDate (message.getDate ());
        cout << " à ";
        displayHour (message.getDate ());
        cout << endl;
        message.getSender ()->getFriend (5);

    } // testPrivateMessage()

M2103-TP9-EXO-4-Corrigé

    class PrivateMessage : public Message
    {
        User * mySender;
      public :
        PrivateMessage (const string & content, User * sender)
            : Message (content), mySender (sender) {}

        User * getSender (void) const { return mySender; }

    }; // PrivateMessage