44
String char * std::string <string> <cstring> pr ’\0’ strcat,strcmp,strcpy... <cstring> length newcopy strutil.h 1 #ifndef SRC _ STRUTIL _ H _ 2 #define SRC _ STRUTIL _ H _ 3 4 #include <cstring> 5 6 namespace pr { 7 8 size _ t length (const char * str); 9 10 char * newcopy (const char * str); 11 12 } 13 14 15 #endif / * SRC _ STRUTIL _ H _ * /

TD 1 : Rappels de programmation C et orientée objet

  • Upload
    others

  • View
    1

  • Download
    0

Embed Size (px)

Citation preview

Page 1: TD 1 : Rappels de programmation C et orientée objet

TD 1 : Rappels de programmation C et orientée objet

Objectifs pédagogiques :

• mémoire, pointeur, classe

• syntaxe générale du c++

• constructeur, destructeur

• opérateurs

Introduction

Dans ce premier TD, nous allons réaliser en C++ une classe String appuyée par une représen-tation interne à l'aide d'une chaîne de caractères du C : un pointeur char * vers un espace mé-moire zéro terminé. Cet exercice est à vocation pédagogique, on préférera en pratique utiliser lastd::string standard du header <string> et les fonctions de manipulation de chaînes du C rangéesdans <cstring>.

Pour être sûr de ne pas entrer en con�it avec d'autres applications, nous utiliserons le namespacepr pour notre implémentation.

1.1 Rappels chaîne du C, const.

Les chaînes de caractères du C, sont représentées par un pointeur de caractère qui cible une zonemémoire contenant la chaîne, et se terminant par un ’\0’.

On souhaite implanter nos propres petites fonctions utilitaires travaillant avec ces chaînes. Cetexercice est à vocation pédagogique, en pratique on préférera utiliser les fonctions standard du C(strcat,strcmp,strcpy...) qui se trouvent dans le header standard <cstring> en C++.

On considère :

• une fonction length rendant une taille pour la chaîne de caractère.

• une fonction newcopy prenant une chaîne de caractère en argument et rendant une nouvellecopie de cette chaîne de caractères.

Question 1. Quel sont donc les signatures de ces fonction ? Où et comment les déclarer, sachantqu'on veut en faire des fonctions utilitaires rangées dans le namespace "pr".

�chier strutil.h

strutil.h

1#ifndef SRC_STRUTIL_H_

2#define SRC_STRUTIL_H_

34#include <cstring>56namespace pr {78size_t length (const char *str);910char * newcopy (const char *str);1112}131415#endif /* SRC_STRUTIL_H_ */

NB : protection double include + namespace.

1

Page 2: TD 1 : Rappels de programmation C et orientée objet

1.1 Rappels chaîne du C, const. TD 1

size_t length (const char *str);

Avec size_t un entier non signé représentant une taille, #include <cstddef> ou #include <cstring>pour le voir. Sa taille est plateforme dépendante, généralement compatible avec la taille d'un pointeur(e.g. size_t prend 8 bytes sur une x64). L'argument est certainement const, on ne souhaite pas lemodi�er.

char * newcopy (const char *str);

Le type de retour peut être const char* si on préfère. L'argument est certainement const, on ne souhaite

pas le modi�er.

Question 2. Donnez deux implantations de ces fonctions. Pour la fonction length, comparerune version utilisant la notation str[i] à une version en pointeurs pur. Pour l'allocation, comparerl'utilisation de l'opérateur new du C++ à malloc. Pour la copie comparer une invocation à memcpyà une boucle. Où placer ce code d'implantation ?

�chier strutil.cpp

strutil.cpp

1#include "strutil.h"23#include <cstdlib>45namespace pr {678// version index9size_t length2 (const char *str) {10size_t ret = 0;11for (int i=0 ; str[i] ; ++i) {12++ret;13}14return ret;15}1617// version pointeurs18size_t length3 (const char *str) {19size_t ret = 0;20for ( ; *str ; ++str) {21++ret;22}23return ret;24}2526// version pointeurs + arithmétique (sans compteur)27size_t length (const char *str) {28const char *cp=str;29for ( ; *cp ; ++cp) {30}31return cp-str;32}333435// version index36char * newcopy2 (const char *src) {37size_t n = length(src);38char * dest = new char[n+1];39

2

Page 3: TD 1 : Rappels de programmation C et orientée objet

1.1 Rappels chaîne du C, const. TD 1

40// avec <= pour attraper le dernier ’\0’41for (size_t i=0; i <= n ; ++i) {42dest[i] = src[i];43}4445return dest;46}4748// version index + malloc49char * newcopy3 (const char *src) {50size_t n = length(src);51// void * malloc (size_t sz)52// pour allouer un tableau, multiplier la taille (sizeof) par le nombre de

cellules53// la signature force un cast pour remonter de void * vers le vrai type voulu54char * dest = (char *) malloc( (n+1)*sizeof(char) );5556// avec <= pour attraper le dernier ’\0’57for (size_t i=0; i <= n ; ++i) {58dest[i] = src[i];59}6061return dest;62}6364// une version pointeurs65// il y a beaucoup de variantes.66char * newcopy4 (const char *src) {67size_t n = length(src);68char * dest = new char[n+1];6970char *cd=dest ;71while (true) {72*cd = *src;73if (! *src) {74break;75}76++cd;77++src;78}79return dest;80}8182// version memcpy83// *toujours* la version à recommander en production84// habite dans #include <cstring>85char * newcopy (const char *src) {86size_t n = length(src);87char * dest = new char[n+1];8889// dest, source, nombre de bytes = n+1 avec le \090memcpy(dest,src,n+1);9192return dest;93}9495}

On n'hésite pas faire des dessins pour la version pointeurs.

3

Page 4: TD 1 : Rappels de programmation C et orientée objet

1.1 Rappels chaîne du C, const. TD 1

(Les ++i au lieu de i++, c'est plus du style qu'autre chose, mais l'opérateur post �xé en général induitun temporaire, pas le pré�xé. Bonne habitude en C++, pour la manipulation e.g. des itérateurs çapeut faire une di�érence. Ce n'est pas important.)

Les tests de �n de boucle etc. sur ∗cp ou cp est un char∗, s'expandent en ∗cp! =′ \0′, mais la plupartdes programmeurs C préfèrent la syntaxe plus compacte. Toute valeur di�érente de 0 est vrai dans lestests.

Le new[] du C++ est mieux typé que malloc : le type et la cardinalité sont donnés. Ici sizeof(char)vaut 1, on le sait, mais c'est la syntaxe générale d'un malloc qui est utilisée.

memcpy est plus e�cace que la boucle, dès qu'on a un peu de distance à copier. La copie est faite parblocs et gérée par l'OS, c'est ce qu'on peut faire de mieux pour copier le contenu de la mémoire d'unendroit à un autre. C'est une fonction essentielle, son API pointeur+ size_t en bytes pour la taille estcaractéristique du C.

La dernière version du code à chaque fois est sans doute la plus proche de strlen/strcpy. Les �vraies�versions intègrent du code pour faire les tests mot par mot (e.g. 4 bytes) au lieu de char par char.

Question 3. Ecrivez un programme dans un �chier exo1.cpp qui construit une copie d'une chaîne"Hello World", puis a�che les deux chaines, leurs addresses, leur longueurs, et sort proprement(pas de fuites mémoire).

�chier exo1.cpp

exo1.cpp

1/*2* exo1.cpp3*4* Created on: Sep 11, 20185* Author: ythierry6*/789#include "strutil.h"1011#include <iostream>1213using namespace pr;14using namespace std;151617int main12 () {18const char * str = "Hello";19char * copy = newcopy(str);2021cout << str << length(str) << endl;22cout << copy << length(copy) << endl;2324// A ajouter dans un deuxieme temps25delete [] copy;2627return 0;28}

Donc les �using� permettent d'alléger la syntaxe, sinon il faut �pr::length� et �std::cout, std::endl�.

On fait �return 0� si pas d'erreur, les codes de retour di�érents de 0 indiquent un problème (conventionUnix).

A la �n du programme, �str� est libéré.

4

Page 5: TD 1 : Rappels de programmation C et orientée objet

1.2 Une classe String TD 1

Les litéraux comme �Hello� sont stockés directement dans le segment de code, ils sont non modi�ables(const char *), et ont une durée de vie égale à celle du programme (sauf s'il vient d'une lib dynammiquechargée par dlopen, et qu'on la dlclose intempestivement.)

Sans le �delete[]� à la �n, on a une fuite mémoire, le �new[]/malloc� de �newcopy� n'a pas de �delete[]/new�symétrique.

Pour la détecter, on utilise valgrind, avec l'option �-leak-check=full�.

Question 4. Expliquez comment compiler séparément ces �chiers puis les assembler (linker) faireun programme exécutable, et l'exécuter.

Invoking: GCC C++ Compiler

g++ -std=c++1y -O0 -g3 -Wall -c -o "src/strutil.o" "../src/strutil.cpp"

g++ -std=c++1y -O0 -g3 -Wall -c -o "src/exo1.o" "../src/exo1.cpp"

Invoking: GCC C++ Linkerg++ -o "td1" ./src/exo1.o ./src/strutil.o

Execution./td1

Valgrindvalgrind --leak-check=full --track-origins=yes ./td1

Donc à la compil, on produit un .o (�chier binaire compilé) par source .cpp, avec les �ags

• -std=c++1y : dialecte utilisé, on pourrait avoir -std=c++11, -std=c++14, -std=c++17 . . . Laversion 1y veut dire >= à c++11 sur notre compilo.

• -O0 : niveau d'optimisation, -O0 pour pouvoir debugger, -O2 ou -O3 en production

• -Wall : active tous les warnings de base, il y a d'autres warnings cela dit (cf -Wextra. . . )

• -c : arrêter la compil sur la production du .o

• -o : output �le, par défaut même nom que le source avec .o

• le �chier .cpp à compiler

En pratique on rajoute des �ags pour que le compilo garde la trace des �chiers (include) dont dépendentle source compilé, il faut reconstruire le .o si n'importe lequel de ces �chiers inclus transitivementchangent.

Au link, on assemble tous les morceaux, et on cherche un main.

• -o : output �le, l'exe qu'on va produire

• tous les .o construits à la compilation

• les bibliothèques statiques libFoo.a

• les bibliothèques dynamiques -lFoo pour libFoo.so/.dylib/.DLL

Pas de dialecte à passer, ni de niveau d'optimisation. Cependant sur compilo récent des options comme-fwhole-program font un link optimisé.

1.2 Une classe String

La gestion manuelle des pointeurs et de la mémoire pose des problèmes :

• accès hors d'une zone allouée (par dépassement, ou par accès à un pointeur non initialisé, oupar deref de nullptr),

5

Page 6: TD 1 : Rappels de programmation C et orientée objet

1.2 Une classe String TD 1

• branchement sur une mémoire non alllouée,

• double désallocation.

La structure de classe permet d'éviter un certain nombre de ces erreurs, en encapsulant ces com-portements, de manière à assurer par exemple que les allocations et désallocations sont cohérentes.Le reste de cet exercice consiste à écrire une classe String qui se comporte �bien�.

Question 5. Dans un �chier �String.h�, dé�nir une classe String minimale munie d'un atributlogeant un pointeur vers une chaîne du C et d'un constructeur permettant de positionner cet attribut.Ajoutez une opération membre �length" qui calcule la longueur de la chaîne. Donnez également lecode de l'implantation de la classe, qu'on placera dans �String.cpp�.

On doit pouvoir s'en servir de la manière suivante :

String str = "Hello";size_t len = str.length();

Donc on avance ligne par ligne dans le programme, et on ajoute ce qu'il faut à la classe String.

Cadre général, on déclare une classe dans le bon namespace :

1#ifndef STRING_H_

2#define STRING_H_

34#include <cstddef> // pour size_t5#include <iosfwd> // pour ostream67// pas de "using" dans les .h8namespace pr {910class String {11// private tant qu’on a rien dit12const char * str;13public:14// tout ce qui suit est public15// un ctor16String (const char * ori);17// taille, membre public et const, i.e. ne modifie pas *this18size_t length() const;1920}; // attention au point virgule pour terminer la déclaration de classe !212223} // fin namespace pr2425#endif /* STRING_H_ */

On note :

• les include, le namespace englobant

• visibilité : private tant qu'on n'a rien dit (accès aux membres de la classe seulement), public = toutle monde voit, protected = membres et membres des classes dérivées (mais du namespace/packagecomme en Java)

• la classe n'a pas de notion de visibilité comme en Java, la notion de visibilité ne s'applique qu'auxmembres des classes C++. Le reste se fait avec des namespace.

• l'attribut est déclaré �const char *�, et sera donc non modi�able; on pourra changer de représenta-tion interne (str = autrechose OK) mais pas modi�er str (str[0] =′ \0 NOK). C'est un choix quiest fait ici, qui nous rapproche des String de Java, immuables. La std::string est au contrairemodi�able.

• l'invocation fournie String str = “Hello”; indique l'existence d'un constructeur prenant un�const char *� en argument.

6

Page 7: TD 1 : Rappels de programmation C et orientée objet

1.2 Une classe String TD 1

• pour le reste, cf les commentaires dans le code ci dessus

Et pour l'implantation :

1#include "String.h"2#include "strutil.h" // pour length et newcopy34#include <iostream>56namespace pr {78String::String (const char * ori) : str(ori) {9}1011size_t String::length () const {12return pr::length(str);13}1415} // fin namespace pr1617#endif /* STRING_H_ */

On note :

• les include, le namespace englobant

• les opérations de String sont rangées dans son namespace, i.e. toute classe dé�nit implicitementun namespace.

• l'opérateur :: permet de quali�er les noms, sans plus de précision on cherche dans le namespacecourant

• �using std;� importe les noms dans le namespace courant; au lieu de �std::cout� on peut écrirejuste �cout�.

• sur le constructeur, on a une liste d'initialisation (entre le ':' et l'accolade ouvrante du ctor).Les attributs n'ont pas de valeur par défaut en C++ (non initialisé par défaut). Cette listed'initialisation peut comporter plusieurs initialisations (séparés par des virgules), on y placeégalement les appels aux constructeurs des classes parentes d'une classe dérivée. En�n (stan-dard c++11), on peut déléguer à un autre constructeur à cet endroit.

• sur length, pas de grosse surprise, on délégue sur la fonction de exo 1. Il faut include �stru-til.h�. Mais si on fait juste �return length(str)�, on a un StackOVer�ow (on va par défaut trouverString::length) ! En quali�ant avec pr:: on tombe sur la bonne fonction de exo1.

Question 6. Ajoutez les mécanismes permettant d'imprimer une String à la manière C++ stan-dard.

La signature standard pour un type MyClass :

ostream & operator << (ostream & os, const MyClass & o)

Donc on instancie ici pour String :

ostream & operator << (ostream & os, const String & s)

L'opérateur est binaire et prend deux arguments, ce qui est à sa gauche, et ce qui est à sa droite. Agauche donc, on s'attends à trouver un �ostream &�, c'est à dire un �ux de sortie, sur lequel on peutécrire. Le �ux en question pourrait être une sortie standard, std::cout ou std::cerr (soit les stdout/stderrdu C++). Mais ça pourrait également être un �ux sur un �chier de sortie (voir <fstream>), ou un �uxdans une zone mémoire (stringstream, déclaré dans <sstream>).

7

Page 8: TD 1 : Rappels de programmation C et orientée objet

1.2 Une classe String TD 1

Les deux arguments sont passés par référence ; le �ux va être modi�é (on va pousser des choses dedans),donc il n'est pas const. Au contraire l'a�chage ne doit pas modi�er la String, on peut donc utiliser uneconst ref.

Par convention, l'opération rend le �ux sur lequel on travaille, de manière à pouvoir chaîner les appels :

std::cout << str << " et " << str.length() << std::endl;

Donc on résoud

std::cout << str

, ça rend un �ostream &� o1, on traite

o1 << " et "

etc. . .

Niveau déclaration, comme l'opérande qui est à gauche est un stream standard, on ne peut pas l'enrichir(i.e. dé�nir un �operator�� membre de ostream). On déclare donc l'opérateur de façon externe, dans lemême namespace que String, mais à côté de la classe.

Le corps de l'opération est assez simple :

1std::ostream & operator << (std::ostream & os, const pr::String & s) {2return os << s.str ;3}

Ce code utilise le fait que les �char *� du C ont déjà un cas particulier dans le standard, plus précisémentajouter un �char *� ou un �const char *� dans un �ux l'interprète comme un string du C. Si on veuta�cher vraiment l'addresse (et pas le contenu de la string) on peut faire :

os << (void *) s.str ;

.

Notons aussi que le standard dé�nit déjà operator� pour tous les types simples et pas mal des types dela lib standard (e.g. std::string).

Le corps de l'opération accède directement à l'attribut �str� de la String, qui est private dans le contextede l'opérateur. Pour résoudre ce problème on ajoute la déclaration �friend�, dans la déclaration de laclasse. Toute fonction déclarée friend peut accéder aux champs privés de la classe.

C'est à la classe de déclarer ses amis (elle rompt ainsi sélectivement son encapsulation), on ne peut passe déclarer ami sans modi�er la classe dont on veut être l'ami.

Question 7. Le code actuellement proposé ne copie pas la chaîne passée en argument. Rappelezen quoi consiste le comportement par défaut qui est généré par le compilateur pour les opérations :

1// dtor2virtual ~String();3// par copie de ori4String(const String &ori);5// mise à jour du contenu6String & operator=(const String & other);

Qu'a�che actuellement par exemple le programme suivant ?

1String getAlphabet() {2char [27] tab;3for (int i=0; i < 26 ; ++i) {

8

Page 9: TD 1 : Rappels de programmation C et orientée objet

1.2 Une classe String TD 1

4tab[i] = ’a’+i;5}6tab[26]=’\0’;7return String(tab);8}9int main() {10String abc = getAlphabet();11std::cout << abc << std::endl;12}

Citer d'autres problèmes que cela peut poser.

Comportement par défaut du dtor = vide (mais invoque implicitement le super destructeur si héritage).

Comportement par défaut du constructeur par copie : a�ecter chacun des champs/attributs par copie.

Comportement par défaut de opérateur = : a�ecter par = tous les attributs de la classe.

Problèmes : la String ne maîtrise pas sa mémoire, elle n'encapsule pas correctement son comportement.

Exemple de problème sur l'exemple, le tableau est stack alloc dans le frame de getAlphabet, donc sefait nettoyer quand on pop le contexte. Notre String se retrouve à pointer qq part (d'illégal) sur la pile,ce qui provoque une faute mémoire quand va essayer de la coller dans cout (car on va lire la mémoirepointée, et même brancher sur son contenu en cherchant des '0').

Donc la faute, c'est qu'on a free la mémoire que je croyais pouvoir adresser. Autres fautes possibles,on me modi�e le contenu pointé (rupture d'encapsulation).

Question 8.Modi�ez la classe String, pour qu'au contraire, elle soit responsable de la mémoirequ'elle pointe. Si l'on se contentait de ne modi�er que le destructeur et le constructeur, expliquezles problèmes que cela pose sur cet exemple.

1int main() {2String abc = "abc";3{4String bcd(abc);5}67std::cout << abc << std::endl;89String def = "def";10def = abc;1112std::cout << abc << " et " << def << std::endl;13}

Pour qu'elle soit responsable de sa mémoire, on va copier à la construction le char * qui m'est passédans une zone nouvellement allouée. Et donc libérer cette mémoire quand on la détruit.

On ajoute donc :

1String::~String() {2delete [] str;3}4String::String(const char *cstr) {5str = newcopy(cstr);6}

9

Page 10: TD 1 : Rappels de programmation C et orientée objet

1.2 Une classe String TD 1

Si on en reste là et qu'on a les versions par défaut du reste, on a des tas de problèmes. (NB : le compilone générera PAS de comportement par défaut si on fournit une des briques : ctor par copie, operator=,dtor. . . ). Donc ici les problèmes ce serait des problèmes de compilation heureusement.

1int main() {2String abc = "abc";// copie donc dans un bloc nouveau, on a fait le ctor3{4String bcd(abc); // ctor par copie par défaut = aliasing ! on pointe la même zone5} // fin de scope de bcd => dtor de bcd invoqué67std::cout << abc << std::endl; // faute mémoire, la zone est déjà free89String def = "def"; // copie donc, on a fait le ctor10def = abc; // fuite mémoire, on perd l’addresse stockée dans def1112std::cout << def << std::endl; // faute mémoire, la zone est déjà free13} // faute mémoire double free sur dtor de abc, triple free avec le dtor de def

Il faut dessiner la mémoire et éxécuter pas à pas pour expliquer.

On corrige avec la classe complètée ainsi (il y a des choses non demandées, corriger les ctor/dtor/oper-ator= etc. . . )

String.h

1#ifndef STRING_H_

2#define STRING_H_

34#include <cstddef> // pour size_t5#include <iosfwd> // pour ostream67// pas de "using" dans les .h89namespace pr {1011class String {12const char * str;1314// déclaration d’accès friend15friend std::ostream & operator << (std::ostream &os, const pr::String & s);1617public:18// ctor (par copie en fin de TD) de cstr19// + déclaration d’une conversion : const char * -> const String &20String(const char *cstr="");21// dtor22virtual ~String();23// par copie de ori24String(const String &ori);25// mise à jour du contenu26String & operator=(const String & other);2728// taille29size_t length() const;30};3132// encore dans namespace pr.3334// déclaration d’une sérialisation par défaut (toString du c++)35std::ostream & operator << (std::ostream &os, const pr::String & s);

10

Page 11: TD 1 : Rappels de programmation C et orientée objet

1.2 Une classe String TD 1

3637} // fin namespace pr3839#endif /* STRING_H_ */

String.cpp

1#include "String.h"2#include "strutil.h" // pour length et newcopy34#include <iostream>56namespace pr {78// par délégation sur le ctor qui prend un const char *9String::String (const String & ori) : String(ori.str) {10}1112String::String(const char *cstr) {13// if (début du TD)14str = cstr;1516// else (fin du TD)17str = newcopy(cstr);18}1920String::~String() {21delete [] str;22}2324String & String::operator=(const String & other) {25if (this != &other) {26delete[] str;27str = newcopy(other.str);28}29return *this;30}3132size_t String::length () const {33return pr::length(str);34}3536std::ostream & operator << (std::ostream & os, const pr::String & s) {37return os << s.str ;38}3940} // namespace pr

Question 9. Quels autres opérateurs pourrait-on o�rir sur String ?

NB : on va les demander en TME.

Opérateurs de comparaison : operator<, operator==

Opérateur de concaténation : operator+

Opérateur de multiplication (par un scalaire) : 2 ∗ “to′′ = “toto′′

Opérateur d'accès indicé : operator[] . . .

11

Page 12: TD 1 : Rappels de programmation C et orientée objet

TME 1 : Programmation, compilation et exécution en C++

Objectifs pédagogiques :

• mise en place

• classe simple

• opérateurs

• compilation, debugger, valgrind

1.1 Plan des séances

Cette UE de programmation avancée suppose une familiarité avec le C et un langage O-O commeJava (syntaxe de base, sémantique), les premières séances sont l'occasion de se remettre à niveau.Cet énoncé contient donc plusieurs encadrés, en gris, qui rappellent les notions clés à appliquer.

Comme le niveau est assez hétérogène, prenez le temps de bien absorber les concepts si nécessaire,nous pouvons prendre un peu de retard. Les TME proposent souvent des extensions dont lesréalisations sont optionelles (bonus) mais qui permettent d'aller un peu plus loin.

Vous soumettrez l'état de votre TME à la �n de chaque séance, en poussant votre code sur un git.La procédure que l'on va mettre en place ce semestre n'étant pas encore opérationnelle pour cetteséance, créez vous un git ou contentez vous de zipper le dossier "src" pour cette séance.

1.2 Prise en main

Eclipse (CDT) : un environnement de développement (IDE) con�guré pour C/C++.Cet environnement graphique gèrera votre projet localement sur votre compte PPTI. Il vous permet-tra d'éditer les sources, de les compiler et de les exécuter. Son utilisation est fortement recommandée(instructions précises dans les supports), mais d'autres outils peuvent aussi être utilisés (NetBeans,Visual Studio Code, . . . ).

Question 1. Lancement d'Eclipse

Attention, plusieurs versions d'Eclipse cohabitent à la PPTI. Il faut utiliser une version pourdéveloppement CDT, qu'on a déployé dans �/usr/local/eclipseCPP/�. La manière la plus sûreest d'ouvrir un terminal et d'y entrer la commande : /usr/local/eclipseCPP/eclipse.Si c'est la première fois que vous utilisez Eclipse, celui-ci vous demande le nom du répertoireworkspace où il placera par défaut vos projets. On recommande d'utiliser un nouveau dossier~/workspacePR comme dossier de workspace qui va loger les projets.

Question 2. Contruire un nouveau projet C++

Démarrer eclipse puis construire un �File->New->C/C++ Project�; on va utiliser le �C++ ManagedBuild� assez facile d'emploi et bien pris en charge par l'IDE comme système de build. On vaconstruire pour commencer un projet hébergeant simplement un éxécutable, prenez le templatefourni �Hello World�. Assurez-vous que la chaîne de compilation sélectionnée est bien �Linux GCC�.Appelez le projet �TME0�. On peut valider les options par défaut sur les autres onglets.

On a à présent un projet avec un main C++ qui a�che un message.

Question 3. Compiler le projet

Sélectionner le menu �Project->Build Project�.

La �console CDT� (un des onglets dans la partie basse de l'IDE) montre les instructions de compi-lation qui ont été faites. On y voit une étape de compilation et une étape de link.

Eclipse a con�guré une version compilée en mode Debug par défaut. Il place les make�le qu'il aengendré et les �chiers produits par la compilation dans un dossier Debug. A priori, on n'édite pasdirectement ces �chiers, mais plutôt les �Properties� du projet (clic droit sur le projet, dernier item).

Il propose aussi une version Release, compilée avec des �ags plus optimisés, utiliser la �èche/tri-angle à côté du marteau (build) dans le ruban d'outils en haut de l'IDE pour basculer sur cette

12

Page 13: TD 1 : Rappels de programmation C et orientée objet

1.2 Prise en main TME 1

con�guration. Si vous ne voyez pas cet outil, assurez vous d'avoir basculé en Persepctive C/C++(avec le bouton dans le coin en haut à droite de eclipse). On va rester en con�guration Debug pourl'instant.

Un nouvel élément �Binaries� est visible, il contient les binaires qu'on vient de construire. On peutclic-droit sur un binaire et faire �Run As. . . ->Local C/C++ application�. Cela lance le programme,dans la console d'eclipse.

Question 4. Ajoutez dans le main un un tableau �tab" de dix entiers que vous remplirez avec lesentiers 0 à 9. A�chez le contenu du tableau.

On constate une bonne qualité du soulignement au cours de la frappe, et l'auto-complétion disponiblesur ctrl-espace.

Certaines fautes ne sont pas soulignées au cours de la frappe, mais seulement si on lance le build com-plet. Par exemple cette erreur d'utilisation d'une variable non initialisée ne sera indiquée qu'aprèsune compilation.

char * a;cout << a ;

Question 5. Découverte du debugger

Recopier le code suivant dans votre main et lancer le programme.

for (size_t i=9; i >= 0 ; i--) {if (tab[i] - tab[i-1] != 1) {cout << "probleme !";}}

Si �probleme� s'a�che, on sait qu'on a un souci. Double-cliquez dans la marge de l'éditeur, sur laligne qui contient l'a�chage de �probleme�, l'outil crée un Breakpoint pour le debug.

Cliquez sur le binaire, mais cette fois-ci faites �Debug As..->Local C++ Application� au lieu de�Run As�. Accepter de basculer en perspective Debug.

La ligne verte indique la position actuelle dans le code, on la fait évoluer en utilisant les outils dansle ruban du haut.

• Continue : poursuit l'exécution jusqu'au prochain breakpoint (qu'on place en double clic dansla marge)

• Step into, Step Over, Step Return : exécution ligne à ligne

Avec le breakpoint positionné sur notre message, avancer jusqu'à l'atteindre, puis inspecter lesvaleurs des variables (à droite). Pour l'a�chage des pointeurs sous forme de tableau, faire un clicdroit sur la variable et demander le mode tableau.

Question 6. Visualisation de l'état sous debugger

Quel est le contenu des cellules du tableau ? Combien vaut �i� ? Modi�er la déclaration du type de�i�, pour que la boucle aie e�ectivemet lieu 10 fois.

c'est un size_t donc il ne sera jamais négatif.

on utilise le type int à la place.

Question 7. Valgrind et détection des fautes mémoire

A présent, sans avoir complétement debuggé le problème, lancer une analyse avec valgrind. Sélec-tionner le binaire, et faire �Pro�ling Tools->Pro�le with Valgrind�. Il doit vous aider à détecter voserreurs (fautes mémoires et fuites mémoire). Les résultats sont visible dans l'onglet Valgrind.

Lire cette page http://valgrind.org/docs/manual/mc-manual.html#mc-manual.errormsgs décrivantles erreurs détectées par cet outil.

13

Page 14: TD 1 : Rappels de programmation C et orientée objet

1.3 Réalisation d'une classe String TME 1

on doit avoir des erreurs de mémoire non initialisée.

Standard C++ dans Eclipse CDT.Par défaut la version du C++ utilisée est celle par défaut de notre compilateur, ce qui est insu�santpour notre usage.Il faut con�gurer le dialecte pour la version "-std=c++1y", soit la plus récente disponible. Malheureuse-ment il faut faire ce réglage dans deux endroits :

• Pour l'éditeur/correction à la volée des erreurs, on règle une préférence globale, valable pour tousles projets d'un workspace donné. Naviguer vers

� "Window -> Preferences -> C/C++ -> Build -> Settings"

� Ouvrir l'onglet "Discovery".

� Sélectionner dans la liste le "CDT GCC Built-in compiler Settings"

� Ajouter un "-std=c++1y" au �ags, de façon à avoir${COMMAND} ${FLAGS} -std=c++1y -E -P -v -dD "${INPUTS}"

• Pour le compilateur à proprement parler, on doit faire un réglage pour chaque projet séparément.En partant d'un clic droit sur le projet, accéder à ses propriétés:

� "Project properties -> C/C++ Build -> Settings"

� Sous le "GCC C++ compiler" on trouve une rubrique "Dialect"

� sélectionner : "-std=c++1y" dans le menu déroulant.

Attention, il faut faire le premier réglage dans tout nouveau workspace, et le deuxième pour chaquenouveau projet.

1.3 Réalisation d'une classe String

Question 8. Dé�nir et tester les fonctions length et newcopy du TD1.

Question 9. Implanter la classe String conformément aux instructions du TD 1.

Au fur et à mesure de la réalisation de la classe, construire un main qui teste les éléments réalisés.S'assurer qu'il ne provoque aucune faute mémoire sous valgrind.

On commencera par réaliser et tester :

• Constructeur par copie de l'argument

• Destructeur qui invoque delete

• Ajout dans un �ux / impression de la String

• Constructeurs par copie, operator= redé�nis

Inclut aussi la réponse aux question suivantes.

String.h

1#ifndef STRING_H_

2#define STRING_H_

34#include <cstddef> // pour size_t5#include <iosfwd> // pour ostream67// pas de "using" dans les .h89namespace pr {1011class String {12const char * str;

14

Page 15: TD 1 : Rappels de programmation C et orientée objet

1.3 Réalisation d'une classe String TME 1

1314// déclaration d’accès friend15friend String operator+ (const String &s1, const String & s2);16friend bool operator== (const String &s1, const String & s2);17friend std::ostream & operator << (std::ostream &os, const pr::String & s);1819public:20// ctor par copie de cstr21// + déclaration d’une conversion : const char * -> const String &22// + ctor par défaut à chaîne vide23String(const char *cstr="");24// dtor25virtual ~String();26// par copie de ori27String(const String &ori);28// mise à jour du contenu29String & operator=(const String & other);3031// accès (non modifiable !)32char operator[] (size_t index) const;33// accès en modification (illégal, on stocke un const char *)34// permettrait de faire : str[0]=’a’;35// char & operator[] (size_t index);3637// taille38size_t length() const;39// égalité logique40bool operator<(const String & o) const;41};4243// encore dans namespace pr.4445// déclaration d’un opérateur + symétrique (non membre)46String operator+ (const String &s1, const String & s2);47// déclaration d’une sérialisation par défaut (toString du c++)48std::ostream & operator << (std::ostream &os, const pr::String & s);49// comparaison d’égalité50bool operator== (const String &s1, const String & s2);5152} // fin namespace pr5354#endif /* STRING_H_ */

String.cpp

1#include "String.h"2#include "strutil.h" // pour length et newcopy34#include <iostream>56namespace pr {789String::~String() {10delete [] str;11}1213String::String (const String & ori) : String(ori.str) {14}

15

Page 16: TD 1 : Rappels de programmation C et orientée objet

1.3 Réalisation d'une classe String TME 1

1516String & String::operator=(const String & other) {17if (this != &other) {18delete[] str;19str = newcopy(other.str);20}21return *this;22}2324String::String(const char *cstr) {25str = newcopy(cstr);26}2728size_t String::length () const {29return pr::length(str);30}3132String operator+ (const String &s1, const String & s2) {33size_t sz1 = s1.length();34size_t sz2 = s2.length();35// + 1 pour le ’\0’36char res [sz1+ sz2 +1];3738if (true) {39// version memcpy40memcpy(res,s1.str,sz1);41// + 1 pour le ’\0’42memcpy(res+sz1,s2.str,sz2+1);43} else {44// version à la main45char * pr = res;46for (const char * p1=s1.str ; *p1 ; ++p1,++pr) {47*pr = *p1;48}49for (const char * p2=s2.str ; *p2 ; ++p2,++pr) {50*pr = *p2;51}52*pr = ’\0’;53}5455return String (res);56}5758bool operator==(const String & a,const String &b) {59return ! compare(a.str,b.str);60}6162bool String::operator<(const String & o) const {63return compare(str,o.str) < 0;64}6566char String::operator[] (size_t index) const {67return str[index];68}6970// Version modifiable, illégale si on stocke un const char *71//char & String::operator[] (size_t index) {72// // Cette ligne induit un cast qui perd le const du char pointé73// return str[index];

16

Page 17: TD 1 : Rappels de programmation C et orientée objet

1.3 Réalisation d'une classe String TME 1

74//}7576std::ostream & operator << (std::ostream & os, const pr::String & s) {77return os << s.str ;78}7980} // namespace pr

Question 10. Ajouter une fonction utilitaire compare aux fonctions utilitaires rangées dans �stru-til.h�. Elle doit se comporter comme la fonction strcmp standard, elle prend deux chaînes a et b duC en argument, et elle rend une valeur négative si a < b, 0 si les deux chaînes sont logiquementégales, et une valeur positive sinon.

Plus précisément, on itére sur les deux chaînes simultanément (avec deux pointeurs) tant que lesvaleurs pointées sont égales et que la première est di�érente de '\0'. On compare ensuite les carac-tères pointés (on peut faire simplement la di�érence de leurs valeurs).

strutil.h

1#ifndef SRC_STRUTIL_H_

2#define SRC_STRUTIL_H_

34#include <cstring>56namespace pr {78size_t length (const char *str);910char * newcopy (const char *str);1112int compare (const char *a, const char * b);1314}151617#endif /* SRC_STRUTIL_H_ */

strutil.cpp

1#include "strutil.h"23#include <cstdlib>45namespace pr {678// version pointeurs + arithmétique (sans compteur)9size_t length (const char *str) {10const char *cp=str;11for ( ; *cp ; ++cp) {12}13return cp-str;14}1516

17

Page 18: TD 1 : Rappels de programmation C et orientée objet

1.4 Utilisation de shared_ptr TME 1

17// version memcpy18// *toujours* la version à recommander en production19// habite dans #include <cstring>20char * newcopy (const char *src) {21size_t n = length(src);22char * dest = new char[n+1];2324// dest, source, nombre de bytes = n+1 avec le \025memcpy(dest,src,n+1);2627return dest;28}293031int compare (const char *cp, const char * cq) {32for ( ; *cp == *cq && *cp; ++cp,++cq) {33}34return *cp - *cq;35}3637}

Question 11. Pour la comparaison entre deux String, on peut proposer soit des opérateurs mem-bres, soit des fonctions extérieures à la classe (plus symétriques).

Ajouter (en s'appuyant sur compare) :

• Un opérateur de comparaison d'égalité bool operator==(const String &a,const String &b)symétrique, déclaré friend et en dehors de la classe String.

• Un opérateur fournisssant une relation d'ordre bool operator<(const String & b) constmem-bre de la classe String.

Question 12. Pour l'accès aux caractères de la String on peut réaliser un opérateur [].

Dé�nissez char operator[](size_t s)const. Peut-on modi�er la String par ce biais ? Quelle seraitla signature d'une variante permettant de modi�er le contenu de la String ? Essayez de la dé�nir,quel problème de typage (const-ness) se pose ?

Question 13. Pour la construction de String plus longues, on peut proposer un String operator+(constString & a, const String &b) symétrique (non membre). L'algorithme consiste à calculer la longueurtotale du résultat, allouer (sur le stack) un tableau de cette taille, et construire une String avec cerésultat.

Question 14. Lancer votre programme de test sous valgrind. Si vous n'avez pas d'erreurs, intro-duisez en délibérément, puis détectez là avec Valgrind, interprétez le message qu'il produit.

• commentez un "delete[]" dans String,

• commentez une copie, i.e. au lieu de faire la copie, contentez de l'a�ecter à l'attribut de laclasse.

1.4 Utilisation de shared_ptr

Cette partie du TME est considérée comme du "bonus" pour ceux qui sont le plus à l'aise ou quisont déjà familiers avec le C++.

Avec les String que l'on a dé�ni, deux String ne peuvent pas partager leur espace de stockage,même si elles ont le même contenu. Comme le contenu de la String est stocké de façon const,partager la représentation des String (en lecture seule donc) ne devrait pas poser de problème. Maisla gestion mémoire est plus complexe, à quel moment libérer une chaîne pointée ? Une solution

18

Page 19: TD 1 : Rappels de programmation C et orientée objet

1.4 Utilisation de shared_ptr TME 1

possible consiste à utiliser un compteur de références, où la chaîne est désallouée quand elle n'estplus référencée, mais n'est pas copiée e.g. quand on a�ecte une String à une autre.

Lisez la documentation de http://www.cplusplus.com/reference/memory/shared_ptr/ et http://www.cplusplus.com/reference/memory/shared_ptr/shared_ptr/.

Question 15. Dans une copie de votre classe String, que l'on appelera StringPtr, substituer auconst char * un std::shared_ptr<const char> qui est dé�ni dans le header <memory>. Mettez àjour la gestion mémoire au sein de la classe (constructeur, destructeur, copie, . . . ) : on ne gardeque le constructeur par copie d'un const char *, le comportement par défaut des autres opérationsest déjà celui qu'on souhaite. Corriger le code pour qu'il compile et fonctionne bien. Pour obtenirle pointeur nu (const char *) à partir d'un shared_ptr<const char> ptr on utilise ptr.get().

StringPtr.h

1#ifndef StringPtr_H_

2#define StringPtr_H_

34#include <cstddef> // pour size_t5#include <iosfwd> // pour ostream6#include <memory>78// pas de "using" dans les .h910namespace pr {1112class StringPtr {13std::shared_ptr<const char> str;1415// déclaration d’accès friend16friend StringPtr operator+ (const StringPtr &s1, const StringPtr & s2);17friend bool operator== (const StringPtr &s1, const StringPtr & s2);18friend std::ostream & operator << (std::ostream &os, const pr::StringPtr & s);1920public:21// ctor par copie de cstr22// + déclaration d’une conversion : const char * -> const StringPtr &23// + ctor par défaut à chaîne vide24StringPtr(const char *cstr="");2526// ELIMINES : shared_ptr est copy et assignable constructible, et gère sa mémoire.27// dtor28// virtual ~StringPtr();29// par copie de ori30// StringPtr(const StringPtr &ori);31// mise à jour du contenu32// StringPtr & operator=(const StringPtr & other);3334// accès (non modifiable !)35char operator[] (size_t index) const;36// accès en modification (illégal, on stocke un const char *)37// permettrait de faire : str[0]=’a’;38// char & operator[] (size_t index);3940// taille41size_t length() const;42// égalité logique43bool operator<(const StringPtr & o) const;44};45

19

Page 20: TD 1 : Rappels de programmation C et orientée objet

1.4 Utilisation de shared_ptr TME 1

46// encore dans namespace pr.4748// déclaration d’un opérateur + symétrique (non membre)49StringPtr operator+ (const StringPtr &s1, const StringPtr & s2);50// déclaration d’une sérialisation par défaut (toStringPtr du c++)51std::ostream & operator << (std::ostream &os, const pr::StringPtr & s);52// comparaison d’égalité53bool operator== (const StringPtr &s1, const StringPtr & s2);5455} // fin namespace pr5657#endif /* StringPtr_H_ */

StringPtr.cpp

1#include "StringPtr.h"2#include "strutil.h" // pour length et newcopy34#include <iostream>56namespace pr {789StringPtr::StringPtr(const char *cstr) : str(newcopy(cstr),[](char*p){ delete[] p;}) {10}1112size_t StringPtr::length () const {13return pr::length(str.get());14}1516StringPtr operator+ (const StringPtr &s1, const StringPtr & s2) {17size_t sz1 = s1.length();18size_t sz2 = s2.length();19// + 1 pour le ’\0’20char res [sz1+ sz2 +1];2122if (true) {23// version memcpy24memcpy(res,s1.str.get(),sz1);25// + 1 pour le ’\0’26memcpy(res+sz1,s2.str.get(),sz2+1);27} else {28// version à la main29char * pr = res;30for (const char * p1=s1.str.get() ; *p1 ; ++p1,++pr) {31*pr = *p1;32}33for (const char * p2=s2.str.get() ; *p2 ; ++p2,++pr) {34*pr = *p2;35}36*pr = ’\0’;37}3839return StringPtr (res);40}4142bool operator==(const StringPtr & a,const StringPtr &b) {43if (a.str.get() == b.str.get()) {44return true;

20

Page 21: TD 1 : Rappels de programmation C et orientée objet

1.4 Utilisation de shared_ptr TME 1

45}46return ! compare(a.str.get(),b.str.get());47}4849bool StringPtr::operator<(const StringPtr & o) const {50if (str == o.str) {51return false;52}53return compare(str.get(),o.str.get()) < 0;54}5556char StringPtr::operator[] (size_t index) const {57return str.get()[index];58}5960// Version modifiable, illégale si on stocke un const char *61//char & StringPtr::operator[] (size_t index) {62// // Cette ligne induit un cast qui perd le const du char pointé63// return str[index];64//}6566std::ostream & operator << (std::ostream & os, const pr::StringPtr & s) {67return os << s.str ;68}6970} // namespace pr

Question 16. Si on lance le programme sous Valgrind on rencontre un problème d'incohérencenew[]/delete, qui gène l'utilisation. Expliquez ce qui se passe. Pour corriger le problème, faites unerecherche sur StackOver�ow pour les mots clés "shared_ptr array C++", on doit y proposer (vuque l'on n'a pas de compilateur C++17 à la PPTI) d'utiliser le constructeur à deux arguments deshared_ptr pour lui passer une fonction à invoquer pour la désallocation. Adopter une des syntaxesproposées, et s'assurer que le problème est résolu.

Le problème e�ectivement c'est que "shared_ptr" libère à l'aide de "delete" alors qu'on alloue avecnew[] les char de la chaîne.

Le standard C++17 résoudra ce problème de shared_ptr, en attendant (sur nos machines) utiliser le con-structeur à deux arguments de shared_ptr pour lui passer une fonction à invoquer pour la désallocation.On pourra lui passer la lambda suivante : [](char *p) { delete[] p; }.

La lambda fournie résout le problème, cf ce SO pour plus de détails : https://stackoverflow.com/questions/13061979/shared-ptr-to-an-array-should-it-be-used

Ca se passe juste dans le ctor de StringPtr dans le corrigé ligne 9 du CPP.

21

Page 22: TD 1 : Rappels de programmation C et orientée objet

TD 2 : Conteneurs, Itérateurs

Objectifs pédagogiques :

• introduction à la lib standard

• gestion mémoire, allocations

• itérateurs du C++, range

• vecteur<T>, liste<T>, map<K,V>

Introduction

Dans ce TD, nous allons réaliser en C++ des classes Vector<T>,List<T>,Map<K,V> o�rant le con-fort habituel de ces conteneurs classiques. Cet exercice est à vocation pédagogique, on préfér-era en pratique utiliser les classes standard, qu'on trouvera dans les header <vector>, <list>,<unordered_map>.

L'objectif de l'exercice est de bien comprendre les conteneurs standards du c++ et leur API.

Pour être sûr de ne pas entrer en con�it avec d'autres applications, nous utiliserons le namespacepr pour notre implémentation.

1.1 Vecteur : stockage contigü.

Un vecteur stocke de façon contigüe en mémoire des données. C'est une des structures de donnéesles plus simples, mais pour cette raison ça reste un bon choix pour de nombreuses applications.

Question 1. Ecrivez une classe Vector<T> en respectant les contraintes suivantes :

• Le vecteur est muni d'un pointeur vers l'espace mémoire alloué pour stocker les données

• Le vecteur est muni d'une taille size, qui indique le remplissage actuel

• Le vecteur a une taille d'allocation, toujours supérieure ou égale à size

• Les objets de type T ajoutés sont copiés dans l'espace mémoire géré par le vecteur

On fournira pour cette classe les operateurs et fonctions suivantes :

• Vector(int size=10) : un constructeur qui prend la taille d'allocation initiale

• Vector() un destructeur

• operator[] dans ses deux variantes (const ou non), pour consulter ou modi�er le contenu.

• size_t size() const la taille actuelle (nombre d'éléments)

• bool empty() const vrai si la taille actuelle est 0

• void push_back (const T& val) : ajoute à la �n du vecteur, peut nécessiter réallocation etcopie.

Vector.h

1#ifndef SRC_VECTOR_H_

2#define SRC_VECTOR_H_

34#include <cstddef>5#include <ostream>67namespace pr {89template <typename T>10class Vector {11size_t alloc_sz;

22

Page 23: TD 1 : Rappels de programmation C et orientée objet

1.1 Vecteur : stockage contigü. TD 2

12T * tab;13size_t sz;1415void ensure_capacity(size_t n) {16if (n >= alloc_sz) {17// double alloc size18alloc_sz *= 2;19T * newtab = new T[alloc_sz];20for (size_t ind=0; ind < sz ; ind++) {21newtab[ind] = tab[ind];22}23delete[] tab;24tab = newtab;25}26}2728public:29Vector(int size=10): alloc_sz(size),sz(0) {30tab = new T [alloc_sz];31}32virtual ~Vector() {33delete [] tab;34}3536const T & operator[] (size_t index) const {37return tab[index];38}39T& operator[] (size_t index) {40return tab[index];41}4243void push_back (const T& val) {44ensure_capacity(sz+1);45tab[sz++]=val;46}4748size_t size() const { return sz ; }4950typedef const T* const_iterator;51typedef T* iterator;5253const_iterator begin() const {54return tab;55}56const_iterator end() const {57return tab+sz;58}59iterator begin() {60return tab;61}62iterator end() {63return tab+sz;64}65};6667template<typename T>68std::ostream & operator<< (std::ostream & os, const Vector<T> & vec) {69os << "[";70typename Vector<T>::const_iterator it = vec.begin();

23

Page 24: TD 1 : Rappels de programmation C et orientée objet

1.2 Liste : stockage par Chainon. TD 2

71while (it != vec.end()) {72os << *it;73++it;74if (it != vec.end()) {75os << ", ";76}77}78os << "]";79return os;80}8182} /* namespace pr */8384#endif /* SRC_VECTOR_H_ */

1.2 Liste : stockage par Chainon.

Une liste simplement chaînée stocke les données dans des chaînons.

Question 2. Ecrivez une classe Liste<T>.

• Une Liste est munie d'un pointeur vers le premier chaînon qui la constitue (ou nullptr si elleest vide)

• La liste ne stocke pas sa taille, il faut la recalculer

• Un Chainon est composé d'un attribut de type T (le data) et d'un pointeur vers le prochainChainon (ou nullptr).

On fournira pour ce cette classe les operateurs et fonctions suivantes :

• List() : un constructeur par défaut pour la liste vide

• List() un destructeur, qui libère tous les chaînons

• size_t size() const la taille actuelle (nombre d'éléments)

• bool empty() const vrai si la liste est vide

• void push_back (const T& val) : ajoute à la �n de la liste

• void push_front (const T& val) : ajoute en tête de la liste

List.h

1#ifndef SRC_LIST_H_

2#define SRC_LIST_H_

34#include <cstddef>5#include <ostream>67namespace pr {89template <typename T>10class List {1112class Chainon {13public :14T data;15Chainon * next;16Chainon (const T & data, Chainon * next=nullptr):data(data),next(next) {};

24

Page 25: TD 1 : Rappels de programmation C et orientée objet

1.2 Liste : stockage par Chainon. TD 2

17};18Chainon * tete;192021public:22List(): tete(nullptr) {23}24virtual ~List() {25for (Chainon * c = tete ; c ; ) {26Chainon * tmp = c->next;27delete c;28c = tmp;29}30}3132const T & operator[] (size_t index) const {33auto it = begin();34for (size_t i=0; i < index ; i++) {35++it;36}37return *it;38}39T& operator[] (size_t index) {40auto it = begin();41for (size_t i=0; i < index ; i++) {42++it;43}44return *it;45}4647void push_back (const T& val) {48if (tete == nullptr) {49tete = new Chainon(val);50} else {51Chainon * fin = tete;52while (fin->next) {53fin = fin->next;54}55fin->next = new Chainon(val);56}57}5859void push_front (const T& val) {60tete = new Chainon(val,tete);61}6263bool empty() const {64return tete == nullptr;65}6667size_t size() const {68size_t sz = 0;69for (auto it = begin() ; it != end() ; ++it,++sz);70return sz;71}7273class ListIte {74Chainon * cur;75public:

25

Page 26: TD 1 : Rappels de programmation C et orientée objet

1.2 Liste : stockage par Chainon. TD 2

76// default ctor = end()77ListIte (Chainon * cur=nullptr) : cur(cur) {}7879T & operator* () const {80return cur->data;81}82T * operator-> () const {83return & cur->data;84}8586ListIte & operator++ () {87cur = cur->next;88return *this;89}90bool operator!= (const ListIte &o) const {91return cur != o.cur;92}93};94typedef ListIte iterator;95class ConstListIte {96const Chainon * cur;97public:98// default ctor = end()99ConstListIte (const Chainon * cur=nullptr) : cur(cur) {}100101const T & operator* () const {102return cur->data;103}104const T * operator-> () const {105return & cur->data;106}107108ConstListIte & operator++ () {109cur = cur->next;110return *this;111}112bool operator!= (const ConstListIte &o) const {113return cur != o.cur;114}115};116117typedef ConstListIte const_iterator;118119const_iterator begin() const {120return tete;121}122const_iterator end() const {123return nullptr;124}125iterator begin() {126return tete;127}128iterator end() {129return nullptr;130}131};132133template<typename T>134std::ostream & operator<< (std::ostream & os, const List<T> & vec) {

26

Page 27: TD 1 : Rappels de programmation C et orientée objet

1.3 Itérateurs. TD 2

135os << "[";136typename List<T>::const_iterator it = vec.begin();137while (it != vec.end()) {138os << *it;139++it;140if (it != vec.end()) {141os << ", ";142}143}144os << "]";145return os;146}147148} /* namespace pr */149150#endif /* SRC_LIST_H_ */

1.3 Itérateurs.

Question 3. Quelles opérations doit fournir un itérateur du C++ ? Comment itérer de façonstandard sur un conteneur ?

Donc l'itération la plus �standard� est sûrement le foreach,

12vector<int> vec;3// on suppose qu’on ajoute des entiers dans vec45// boucle "foreach"6for (int i : vec) {7// body8}910// Expansion C++1111{12for(auto it = vec.begin(),_end = vec.end(); it != _end; ++it)13{14int i = *it;15// body16}17}

On voit qu'il faut :

• begin et end membres du conteneur, fonctions qui rendent des itérateurs

• L'itérateur doit supporter

� la comparaison d'inégalité (operator !=),

� l'opérateur de déréférencement (operator*, parfois confortable d'avoir aussi operator->)

� le préincrément, (operator++)

Question 4. Ajoutez ces mécanismes à votre Liste. En quoi consiste un itérateur ?

27

Page 28: TD 1 : Rappels de programmation C et orientée objet

1.3 Itérateurs. TD 2

Cf le code fourni. On présente la version non const dans un premier temps.

L'itérateur pointe naturellement un chainon cur ; operator* rend le data stocké dans le chaînon, etoperator++ fait cur = cur->next.

Begin rend un itérateur cur=tete de liste, end rend un itérateur cur=nullptr.

Question 5. Ajoutez ces mécanismes à votre Vector. En quoi consiste un itérateur ?

Cf le code fourni. On présente la version non const dans un premier temps.

L'itérateur est tout bêtement un pointeur dans le tableau, pas besoin de dé�nir de classe. Un bontypedef (dans la partie public) est cependant une bonne idée, pour que Vector<T>::iterator soit dé�nipour les clients.

Begin rend un pointeur sur le début de la zone du tableau, end rend un pointeur un au delà du dernierélément.

Question 6. Expliquez la di�érence entre les iterator et const_iterator. Expliquez comment ilfaut modi�er Vector et List pour permettre des itérations const.

Les signatures const/non const interfèrent pas mal, on est forcés de dupliquer un peu le code desitérateurs.

Sur Vector c'est juste rajouter les begin/end const essentiellement.

Sur Liste il faut aussi dupliquer l'itérateur.

Question 7. Ecrivez une fonction qui a�che le contenu d'un Vector. Comment la modi�er à l'aidede template pour permettre d'a�cher également une Liste ? Utilisez la fonction dans un main.

Une version de �base�.

main.cpp

1#include "Vector.h"2#include <iostream>345using namespace std;6using namespace pr;789void show_const2 (const Vector<int> & vec) {10std::cout << "const:" << std::endl;11for (const auto & val : vec) {12std::cout << val;13}14std::cout << std::endl;15}1617void show2 (Vector<int> & vec) {18std::cout << "avant:" << std::endl;19for (auto & val : vec) {20std::cout << val << " ";21++val;22}23std::cout << "apres:" << std::endl;24show_const2(vec);

28

Page 29: TD 1 : Rappels de programmation C et orientée objet

1.3 Itérateurs. TD 2

25std::cout << std::endl;26}272829int main2 () {30Vector<int> vec (5);3132for (int i=0; i < 10 ; i++) {33vec.push_back(i);34}3536std::cout << vec << std::endl;373839show_const2(vec);4041show2(vec);4243std::cout << vec[4] << std::endl;4445return 0;46}

version templati�ée :

main.cpp

1#include "List.h"2#include "Vector.h"3#include <iostream>456using namespace std;7using namespace pr;8910template< template<typename> class Container, typename T>11void show_const (const Container<T> & vec) {12std::cout << "const:" << std::endl;13for (const auto & val : vec) {14std::cout << val;15}16std::cout << std::endl;17}181920template< template<typename> class Container, typename T>21void show (Container<T> & vec) {22std::cout << "avant:" << std::endl;23for (auto & val : vec) {24std::cout << val << " ";25++val;26}27std::cout << "apres:" << std::endl;28show_const(vec);29std::cout << std::endl;30}313233

29

Page 30: TD 1 : Rappels de programmation C et orientée objet

1.3 Itérateurs. TD 2

34template< template<typename> class Container, typename T>35int test (Container<T> & vec) {3637for (int i=0; i < 15 ; i++) {38vec.push_back(i);39}4041std::cout << vec << "size " << vec.size() <<std::endl;424344show_const(vec);4546show(vec);4748std::cout << vec[4] << std::endl;4950return 0;51}5253int main () {54Vector<int> vec;55test(vec);56List<int> list;57test(list);58}

30

Page 31: TD 1 : Rappels de programmation C et orientée objet

TME 2 : Conteneurs, Itérateurs, Lib Standard

Objectifs pédagogiques :

• conteneurs

• itérateurs

• map

1.1 Liste, Vecteur

Question 1. En suivant les indications du TD 2, implanter les classes génériques Liste<T> etVector<T> et leurs itérateurs. Elles doivent pouvoir s'utiliser avec ce main (générique) :

main_template.cpp

1#include "List.h"2#include "Vector.h"3#include <iostream>45using namespace std;6using namespace pr;789template< template<typename> class Container, typename T>10void show_const (const Container<T> & vec) {11std::cout << "const:" << std::endl;12for (const auto & val : vec) {13std::cout << val;14}15std::cout << std::endl;16}171819template< template<typename> class Container, typename T>20void show (Container<T> & vec) {21std::cout << "avant:" << std::endl;22for (auto & val : vec) {23std::cout << val << " ";24++val;25}26std::cout << "apres:" << std::endl;27show_const(vec);28std::cout << std::endl;29}30313233template< template<typename> class Container, typename T>34int test (Container<T> & vec) {3536for (int i=0; i < 15 ; i++) {37vec.push_back(i);38}3940std::cout << vec << "size " << vec.size() <<std::endl;4142show_const(vec);43show(vec);4445return 0;46}

31

Page 32: TD 1 : Rappels de programmation C et orientée objet

1.1 Liste, Vecteur TME 2

4748int main_t () {49Vector<int> vec;50test(vec);51std::cout << vec[4] << std::endl;52vec[4]++;53std::cout << vec[4] << std::endl;5455List<int> list;56test(list);57return 0;58}

C'est les même qu'au TD.

Vector.h

1#ifndef SRC_VECTOR_H_

2#define SRC_VECTOR_H_

34#include <cstddef>5#include <ostream>67namespace pr {89template <typename T>10class Vector {11size_t alloc_sz;12T * tab;13size_t sz;1415void ensure_capacity(size_t n) {16if (n >= alloc_sz) {17// double alloc size18alloc_sz *= 2;19T * newtab = new T[alloc_sz];20for (size_t ind=0; ind < sz ; ind++) {21newtab[ind] = tab[ind];22}23delete[] tab;24tab = newtab;25}26}2728public:29Vector(int size=10): alloc_sz(size),sz(0) {30tab = new T [alloc_sz];31}32virtual ~Vector() {33delete [] tab;34}3536const T & operator[] (size_t index) const {37return tab[index];38}39T& operator[] (size_t index) {40return tab[index];41}42

32

Page 33: TD 1 : Rappels de programmation C et orientée objet

1.1 Liste, Vecteur TME 2

43void push_back (const T& val) {44ensure_capacity(sz+1);45tab[sz++]=val;46}4748size_t size() const { return sz ; }4950typedef const T* const_iterator;51typedef T* iterator;5253const_iterator begin() const {54return tab;55}56const_iterator end() const {57return tab+sz;58}59iterator begin() {60return tab;61}62iterator end() {63return tab+sz;64}65};6667template<typename T>68std::ostream & operator<< (std::ostream & os, const Vector<T> & vec) {69os << "[";70typename Vector<T>::const_iterator it = vec.begin();71while (it != vec.end()) {72os << *it;73++it;74if (it != vec.end()) {75os << ", ";76}77}78os << "]";79return os;80}8182} /* namespace pr */8384#endif /* SRC_VECTOR_H_ */

List.h

1#ifndef SRC_LIST_H_

2#define SRC_LIST_H_

34#include <cstddef>5#include <ostream>67namespace pr {89template <typename T>10class List {1112class Chainon {13public :14T data;

33

Page 34: TD 1 : Rappels de programmation C et orientée objet

1.1 Liste, Vecteur TME 2

15Chainon * next;16Chainon (const T & data, Chainon * next=nullptr):data(data),next(next) {};17};18Chainon * tete;192021public:22List(): tete(nullptr) {23}24virtual ~List() {25for (Chainon * c = tete ; c ; ) {26Chainon * tmp = c->next;27delete c;28c = tmp;29}30}3132void push_back (const T& val) {33if (tete == nullptr) {34tete = new Chainon(val);35} else {36Chainon * fin = tete;37while (fin->next) {38fin = fin->next;39}40fin->next = new Chainon(val);41}42}4344void push_front (const T& val) {45tete = new Chainon(val,tete);46}4748bool empty() const {49return tete == nullptr;50}5152size_t size() const {53size_t sz = 0;54for (auto it = begin() ; it != end() ; ++it,++sz);55return sz;56}5758class ListIte {59Chainon * cur;60public:61// default ctor = end()62ListIte (Chainon * cur=nullptr) : cur(cur) {}6364T & operator* () const {65return cur->data;66}67T * operator-> () const {68return & cur->data;69}7071ListIte & operator++ () {72cur = cur->next;73return *this;

34

Page 35: TD 1 : Rappels de programmation C et orientée objet

1.1 Liste, Vecteur TME 2

74}75bool operator!= (const ListIte &o) const {76return cur != o.cur;77}78};79typedef ListIte iterator;80class ConstListIte {81const Chainon * cur;82public:83// default ctor = end()84ConstListIte (const Chainon * cur=nullptr) : cur(cur) {}8586const T & operator* () const {87return cur->data;88}89const T * operator-> () const {90return & cur->data;91}9293ConstListIte & operator++ () {94cur = cur->next;95return *this;96}97bool operator!= (const ConstListIte &o) const {98return cur != o.cur;99}100};101102typedef ConstListIte const_iterator;103104const_iterator begin() const {105return tete;106}107const_iterator end() const {108return nullptr;109}110iterator begin() {111return tete;112}113iterator end() {114return nullptr;115}116};117118template<typename T>119std::ostream & operator<< (std::ostream & os, const List<T> & vec) {120os << "[";121typename List<T>::const_iterator it = vec.begin();122while (it != vec.end()) {123os << *it;124++it;125if (it != vec.end()) {126os << ", ";127}128}129os << "]";130return os;131}132

35

Page 36: TD 1 : Rappels de programmation C et orientée objet

1.2 std::vector, std::pair TME 2

133} /* namespace pr */134135#endif /* SRC_LIST_H_ */

1.2 std::vector, std::pair

On part du programme suivant, qui essaie de compter combien de mots di�érents sont utilisés dansun livre.

Compte le nombre mots différents

12#include <iostream>3#include <fstream>4#include <regex>5#include <algorithm>6#include <vector>7#include <chrono>89using namespace std;1011int main_p () {12vector<string> words;13words.reserve(5000);1415ifstream input = ifstream("/tmp/WarAndPeace.txt");1617std::chrono::steady_clock::time_point start = std::chrono::steady_clock::now();18std::cout << "Parsing War and Peace" << endl;1920std::string s;21// une regex qui reconnait les caractères anormaux (négation des lettres)22regex re( R"([^a-zA-Z])");23while (input >> s) {24// élimine la ponctuation et les caractères spéciaux25s = regex_replace ( s, re, "");26// passe en lowercase27std::transform(s.begin(),s.end(),s.begin(),::tolower);2829// cherchons si le mot est déjà présent30auto it = words.begin();31while (it != words.end()) {32if (*it == s) {33// trouvé34break;35}36++it;37}38if (it != words.end()) {39// déjà trouvé40continue;41} else {42words.push_back(s);43}44}45input.close();4647std::cout << "Finished Parsing War and Peace" << endl;

36

Page 37: TD 1 : Rappels de programmation C et orientée objet

1.3 Table de Hash TME 2

4849std::chrono::steady_clock::time_point end = std::chrono::steady_clock::now();50std::cout << "Parsing with took "51<< std::chrono::duration_cast<std::chrono::milliseconds>(end - start).count()52<< "ms.\n";5354std::cout << "Found a total of " << words.size() << " words." << std::endl;5556return 0;57}

Question 2. Exécutez le programme sur le �chier WarAndPeace.txt fourni. Combien y a-t-il demots di�érents ?

Sur ma machine, j'ai

Parsing War and PeaceFinished Parsing War and PeaceParsing with took 17751ms.Found a total of 20333 words.

Question 3.Modi�ez le programme pour qu'il calcule le nombre d'occurrences de chaque mot.Pour cela, on adaptera le code pour utiliser un vecteur qui stocke des pair<string,int> au lieu destocker juste des string. A�cher le nombre d'occurrences des mots �war�, �peace� et �toto�.

On doit trouver :

Frequency : war:298Frequency : peace:114Word :toto not found.

Question 4. Que penser de la complexité algorithmique de ce programme ? Quelles autres struc-tures de données de la lib standard aurait-on pu utiliser ?

Oui, ben c'est pas terrible, on est en O(N2), pour chaque mot rencontré, au pire des cas on scanne tousles mots qu'on a déjà vus.

Le hash (unordered_map) o�re du O(N) si on a pas trop de collisions, et pas besoin de réindexer(dimensionnement initial adéquat).

1.3 Table de Hash

On souhaite à présent implanter une table de hash assez simple, en appui sur les classes Liste<T>et Vector<T> qu'on a développé plus haut.

On rappelle les principes d'une table de hash :

• La table stocke un vecteur de taille relativement grande appelé buckets.

• Dans chaque case du vecteur, on trouve une liste de paires (clé,valeur)

• Pour rechercher une entrée à partir d'une clé k, on hash la clé, ce qui rend un entier hash(k)dont la valeur dépend du contenu de la clé.

• On cherche dans le bucket d'indice hash(k) % buckets.size(), un élément de la liste dont laclé serait égale (au sens de ==) avec la clé k recherchée.

37

Page 38: TD 1 : Rappels de programmation C et orientée objet

1.3 Table de Hash TME 2

• Si on le trouve, on peut exhiber la valeur qui lui est associée, sinon c'est qu'il n'est pas présentdans la table.

Question 5. Ecrire une classe générique HashTable<K,V> où K est un paramètre qui donne le typede la clé, et V donne le type des valeurs.

On pourra dans l'ordre :

• dé�nir le cadre de la classe, ses paramètres génériques

• dé�nir un type Entry pour contenir les paires clés valeur ; les clés sont stockées de façon const,pas les valeurs

• ajouter un attribut typé Vector< List< Entry > > dans la classe

• dé�nir un constructeur prenant une taille, qui initialise le vecteur avec des listes vides danschaque bucket

Ensuite, dé�nir un itérateur (version non const), son opération de déréférencement, sa comparaison,son incrément.

• il porte une réference vers la table buckets

• il porte un indice ou un itérateur vit dans cette table, désignant la liste (bucket) qu'il exploreactuellement

• il porte un itérateur lit de liste, qui pointe l'élément courant dans la liste que désigne l'itérateur

• pour l'incrémenter, on incrémente d'abord lit, si l'on est au bout de la liste, on se décale survit à la recherche d'une case non vide. lit devient alors la tête de cette liste.

• pour la comparaison d'égalité, on peut se contenter de comparer les itérateurs lit

• le déréférencement rend une Entry, celle qui est pointée par lit

En�n dé�nir les méthodes utilisant ces itérateurs :

• Dé�nir les opérations begin() et end() de la table de hash.

• Dé�nir iterator find (const K & key) qui rend un itérateur correctement positionné, ouend() si la clé n'est pas trouvée. Pour calculer la valeur de hash de l'objet key, on utiliserasize_t h = std::hash<K>()(key);. Cette fonction ne doit pas itérer toute la table, seulementle �bucket� approprié.

• Dé�nir std::pair<iterator,bool> insert (const Entry & entry) qui essaie d'insérer la paireclé valeur fournie dans la table.

� Si la clé était déjà présente dans la table, le booléen rendu est faux, et l'itérateur pointel'entrée qui existait déjà dans la table (qui n'a pas été modi�ée).

� Si la clé n'était pas encore présente, le booléen rendu est vrai, et l'itérateur pointe l'entréequi vient d'être ajoutée.

HashMap.h

1#ifndef SRC_HASHMAP_H_

2#define SRC_HASHMAP_H_

34#include <cstddef>5#include <ostream>67#include "List.h"8#include "Vector.h"910namespace pr {1112template <typename K, typename V>13class HashMap {1415public:

38

Page 39: TD 1 : Rappels de programmation C et orientée objet

1.3 Table de Hash TME 2

16class Entry {17public :18const K key;19V value;20};21private :2223typedef Vector<List<Entry> > buckets_t;24// storage for buckets table25buckets_t buckets;26// total number of entries27size_t sz;2829public:30HashMap(size_t size): buckets(size),sz(0) {31for (size_t i =0; i < size ; i++) {32buckets.push_back(List<Entry>());33}34}3536class HashMapIte {37buckets_t & table;38size_t index;39typename List<Entry>::iterator cur;40public:41// begin() ctor42HashMapIte (buckets_t & table) :table(table),index(0) {43cur = table[0].end();44for (index=0; index < table.size() ; index++) {45if (! table[index].empty()) {46cur = table[index].begin();47break;48}49}50}5152HashMapIte (buckets_t & table, size_t index, typename List<Entry>::iterator

cur) : table(table),index(index),cur(cur) {}5354Entry & operator* () const {55return *cur;56}57Entry * operator-> () const {58return & (*cur);59}60HashMapIte & operator++ () {61++cur;62while (index < table.size() && ! (cur != table[index].end())) {63++index;64cur = table[index].begin();65}66return *this;67}68bool operator!= (const HashMapIte &o) const {69return cur != o.cur;70}71};72typedef HashMapIte iterator;73

39

Page 40: TD 1 : Rappels de programmation C et orientée objet

1.3 Table de Hash TME 2

74iterator begin() {75return HashMapIte(buckets);76}77iterator end() {78return HashMapIte(buckets,buckets.size(),nullptr);79}8081iterator find (const K & key) {82size_t h = std::hash<K>()(key);83size_t target = h % buckets.size();84for (auto it = buckets[target].begin() ; it != buckets[target].end() ; ++ it

) {85if (it->key == key) {86// Found it !87return iterator(buckets,target,it);88}89}90// miss91return end();92}9394std::pair<iterator,bool> insert (const Entry & entry) {95size_t h = std::hash<K>()(entry.key);96size_t target = h % buckets.size();9798for (auto it = buckets[target].begin() ; it != buckets[target].end() ; ++ it

) {99if (it->key == entry.key) {100// Found it !101return std::make_pair(iterator(buckets,target,it),false);102}103}104// not found, do insertion (in front)105buckets[target].push_front(entry);106++sz;107return std::make_pair(iterator(buckets,target,buckets[target].begin()),true)

;108}109110size_t size() const { return sz ; }111112};113114template<typename K, typename V>115std::ostream & operator<< (std::ostream & os, HashMap<K,V> & vec) {116os << "[";117typename HashMap<K,V>::iterator it = vec.begin();118int perline = 0;119while (it != vec.end()) {120os << it->key << " -> " << it->value ;121++it;122if (it != vec.end()) {123os << ", ";124}125perline++;126if (perline == 10) {127perline =0;128os << std::endl;129}

40

Page 41: TD 1 : Rappels de programmation C et orientée objet

1.4 Mots les plus fréquents TME 2

130}131os << "]";132return os;133}134135} /* namespace pr */136137#endif /* SRC_HASH_H_ */

1.4 Mots les plus fréquents

Question 6. Utiliser une table de hash associant des entiers (le nombre d'occurrence) aux mots,et reprendre les question où l'on calculait de nombre d'occurrences de mots avec cette nouvellestructure de donnée.

On a beau avoir fait une hashtable un peu naive, on doit avoir divisé les temps par 10 à 100.

Corrigé de cette série de questions dans le code à la �n.

Question 7. Après avoir chargé le livre, initialiser un std::vector<pair<string,int> >, par copiedes entrées dans la table de hash. On pourra utiliser le contructeur par copie d'une range : vector(InputIterator first, InputIterator last).

Question 8. Ensuite trier ce vecteur par nombre d'occurrences décroissantes à l'aide de std::sort.On lui passe les itérateurs de début et �n de la zone à trier, et un prédicat binaire. Voir l'exemplesuivant.

exemple sort et lambda

1#include <vector>2#include <string>3#include <algorithm>45class Etu {6public :7std::string nom;8int note;9};1011int main_sort () {12std::vector<Etu> etus ;13// plein de push_back de nouveaux étudiants dans le désordre1415// par ordre alphabétique de noms croissants16std::sort(etus.begin(), etus.end(), [] (const Etu & a, const Etu & b) { return a.nom <

b.nom ;});17// par notes décroissantes18std::sort(etus.begin(), etus.end(), [] (const Etu & a, const Etu & b) { return a.note

> b.note ;});19return 0;20}

41

Page 42: TD 1 : Rappels de programmation C et orientée objet

1.4 Mots les plus fréquents TME 2

main_hash.h

12#include "HashMap.h"3#include <iostream>4#include <fstream>5#include <regex>6#include <algorithm>7#include <vector>8#include <chrono>910using namespace std;11using namespace pr;121314void printFrequency (const string & word, HashMap<string,int> & map) {15auto it = map.find(word);16if (it != map.end()) {17std::cout << "Frequency : " << it->key << ":" << it->value << endl;18} else {19std::cout << "Word :"<< word << " not found." << endl;20}2122}2324int main () {2526HashMap<std::string,int> map(2<<18); // 256 k word capacity27ifstream input = ifstream("/tmp/WarAndPeace.txt");2829std::chrono::steady_clock::time_point start = std::chrono::steady_clock::now();30std::cout << "Parsing War and Peace" << endl;31std::string s;32regex re( R"([^a-zA-Z])");33while (input >> s) {34s = regex_replace ( s, re, "");35std::transform(s.begin(),s.end(),s.begin(),::tolower);36auto ret = map.insert({s,1});37if (! ret.second) {38(*ret.first).value++;39}40}41input.close();42// std::cout << "Counts :" << map << endl;43std::cout << "Finished Parsing War and Peace" << endl;4445std::chrono::steady_clock::time_point end = std::chrono::steady_clock::now();46std::cout << "Parsing with hash took "47<< std::chrono::duration_cast<std::chrono::milliseconds>(end - start).count()48<< "ms.\n";4950start = std::chrono::steady_clock::now();51std::cout << "Parsing War and Peace" << endl;52vector<pair<string,int> > counts;53counts.reserve(5000);54input = ifstream("/tmp/WarAndPeace.txt");55while (input >> s) {56s = regex_replace ( s, re, "");57std::transform(s.begin(),s.end(),s.begin(),::tolower);58auto it = counts.begin();

42

Page 43: TD 1 : Rappels de programmation C et orientée objet

1.4 Mots les plus fréquents TME 2

59while (it != counts.end()) {60if (it->first == s) {61break;62}63++it;64}65if (it != counts.end()) {66it->second++;67} else {68counts.push_back(make_pair(s,1));69}70}71input.close();72end = std::chrono::steady_clock::now();73std::cout << "Parsing with pair<string,int> took "74<< std::chrono::duration_cast<std::chrono::milliseconds>(end - start).

count()75<< "ms.\n";7677printFrequency("war",map);78printFrequency("peace",map);79printFrequency("toto",map);8081start = std::chrono::steady_clock::now();82std::cout << "Most frequent" << endl;83std::vector<pr::HashMap<std::string,int>::Entry * > entries;84entries.reserve(map.size());85for (auto it = map.begin() ; it != map.end() ; ++it) {86entries.push_back(& (*it) );87}88std::sort(entries.begin(),entries.end(), [] (auto a,auto b) { return a->value > b

->value ;});89int line=0;90for (const auto & val : entries) {91std::cout << val->key << "->" << val->value << " ,";92if (++line % 10 == 0) {93std::cout << endl;94}95if (line == 100) {96break;97}98}99std::cout << std::endl;100end = std::chrono::steady_clock::now();101std::cout << "Sorting by frequency took "102<< std::chrono::duration_cast<std::chrono::microseconds>(end - start).count()103<< "us.\n";104105return 0;106}

Question 9. Si la table de hash est trop pleine, on aura beaucoup de collisions de hash, c'est à direque les listes stockées dans chaque bucket deviennent de plus en plus longues. Ecrivez une fonctiongrow dans votre table de hash, qui double la taille du vecteur de buckets, et réinsère les élémentsdans le résultat. Quelle est la complexité de cette réindexation ?

43

Page 44: TD 1 : Rappels de programmation C et orientée objet

1.4 Mots les plus fréquents TME 2

Donc on initialise une nouvelle table, on itère sur this, on insère tout ce qu'on rencontre dans la nouvelletable (avec insert).

Je ne l'ai pas codé désolé, c'est une question bonus.

Bilan : il faut recalculer tous les hash et les modulo.

Vu qu'on sait ce qu'on est en train de faire, et que la propriété de clé est présente initialement, on peutse contenter de push_front sur le bon bucket, et ne jamais itérer sur une liste dans l'algo.

Ca reste une opération chère, en mémoire (le double d'une taille déjà calculé pour être un peu grosse)

et en temps (rehasher toutes les clés).

44