Les Piles Et Les Files en C
Les Piles Et Les Files en C
Les Piles Et Les Files en C
en C++
Par Xavinou
www.openclassrooms.com
2/16
Sommaire
Sommaire ........................................................................................................................................... 2
Les piles et les files en C++ ............................................................................................................... 3
Principe .............................................................................................................................................................................. 3
Les piles ...................................................................................................................................................................................................................... 3
Les files ....................................................................................................................................................................................................................... 3
La STL ......................................................................................................................................................................................................................... 3
Les piles ............................................................................................................................................................................ 3
La hauteur de la pile .................................................................................................................................................................................................... 4
L'état de la pile ............................................................................................................................................................................................................ 4
Ajouter (empiler) un élément dans la pile .................................................................................................................................................................... 5
Accéder au dernier élément de la pile ......................................................................................................................................................................... 5
Supprimer (dépiler) le dernier élément de la pile ........................................................................................................................................................ 5
Passage de paramètres .............................................................................................................................................................................................. 6
Les files ............................................................................................................................................................................. 8
La taille de la file ......................................................................................................................................................................................................... 9
L'état de la file ............................................................................................................................................................................................................. 9
Ajouter un nouvel élément .......................................................................................................................................................................................... 9
Supprimer la première donnée .................................................................................................................................................................................... 9
Accéder au premier et au dernier élément .................................................................................................................................................................. 9
Exemple d'utilisation ........................................................................................................................................................ 11
Les EPF (Expressions Post-Fixées) .......................................................................................................................................................................... 12
Partager ..................................................................................................................................................................................................................... 15
www.openclassrooms.com
Sommaire 3/16
Par Xavinou
Bonjour à tous !
Dans ce tutoriel, vous allez apprendre à vous servir des piles ainsi que des files en C++.
Avant de vous lancer dans la lecture, vous pouvez lire le tutoriel d'Octal sur les piles et les files en C, pour voir le principe.
L'avantage en C++, c'est que l'implémentation et l'utilisation ont été considérablement simplifiées !
Sommaire du tutoriel :
Principe
Les piles
Les files
Exemple d'utilisation
Principe
Avant de commencer à coder, voyons le principe de ces structures de données.
Les piles
Tous les jours, on manipule des piles (et pas seulement dans la télécommande !).
Prenons un exemple : une pile d'assiettes à laver. On les met les unes sur les autres et pour les laver, on prend celle du dessus. Un
deuxième exemple : une pile de dossiers sur un bureau. On les met les uns au-dessus des autres et on retire le premier, donc le
dernier arrivé. Les piles répondent à ce principe : dernier arrivé, premier servi, aussi connues sous le nom de LIFO pour Last In,
F irst Out.
Les files
Pour les files, c'est exactement le contraire ! Un exemple tout simple : une file d'attente. Le premier arrivé est le premier à sortir (à
être servi), aussi connues sous le nom de FIFO pour F irst In, F irst Out.
La STL
STL pour Standard Template Library. Les deux structures de données de ce tuto en font partie. Pour résumer, la STL est un
ensemble de structures pouvant contenir n'importe quel type de données. Cela veut donc dire que, dans les piles et les files, on
peut stocker tout type de données (entiers, réels, chaînes de caractères, ...) mais un seul type par structure : si je déclare une pile
d'entiers, je ne pourrais pas empiler des chaînes de caractères !
www.openclassrooms.com
Les piles et les files en C++ 4/16
Les piles
Bien, voyons en détail les piles !
Code : C++
#include <stack>
using namespace std;
Code : C++
Entre les chevrons (< >), il faut mettre le type de pile voulu. Si l'on désire une pile d'entiers, on met int ; pour une pile de chaînes,
on met string, ...
Voyons maintenant les différentes manipulations des piles.
La hauteur de la pile
Pour pouvoir boucler sur une pile, il faut connaître sa hauteur (ou sa taille). On utilise la fonction nomPile.size() qui renvoie le
nombre d'éléments de la pile.
Code : C++
stack<int> i;
cout << "Le nombre d'éléments de la pile est : " << i.size() <<
endl;
Ce code affichera :
Code : Console
La taille est zéro car lorsqu'une pile est initialisée, elle est initialisée en tant que pile vide, donc elle ne contient aucun élément.
L'état de la pile
Si on ne veut pas connaître le nombre d'éléments présents dans la pile, mais juste savoir si elle est vide ou non, on utilise la
fonction nomPile.empty() qui renvoie un booléen qui vaut true si la pile est vide et false sinon.
www.openclassrooms.com
Les piles et les files en C++ 5/16
Code : C++
stack<int> i;
cout << "Etat de la pile : ";
if(i.empty())
cout << "vide" << endl;
else
cout << "pas vide" << endl;
Pour empiler un élément, on utilise la fonction nomPile.push(elt) qui prend en paramètre l'élément à empiler.
Code : C++
Maintenant, si on veut accéder aux éléments de la pile, il faut obligatoirement prendre le dernier empilé, donc celui qui se trouve
au sommet. Pour récupérer cet élément, on utilise la fonction nomPile.top().
Code : C++
stack<int> i;
i.push(2);
i.push(12);
Comme vous le voyez, il est impossible d'afficher le second élément. Il faut donc supprimer le premier.
www.openclassrooms.com
Les piles et les files en C++ 6/16
Code : C++
stack<int> i;
i.push(2);
i.push(3);
cout << "Le premier élément est : " << i.top() << endl; // accès au
dernier élément
i.pop(); // suppression du dernier élément
cout << "Le deuxième élément est : " << i.top() << endl; // accès
au deuxième élément
i.pop(); // suppression du dernier élément ; maintenant la pile
est vide !
cout << "Nombre d'éléments dans la pile : " << i.size() << endl;
Passage de paramètres
Implicitement, le passage de paramètres se fait par valeurs. Une pile passée en paramètre d'un sous-programme est donc
recopiée. Pour des raisons de performances, une pile en entrée d'une procédure ou d'une fonction est passée par référence en
ajoutant un "et commercial" (&) avant le nom du paramètre (exemple : stack<int>& p).
Je vais maintenant vous donner deux exemples sur le fonctionnement de la pile en utilisant toutes les fonctions que l'on a vues.
Le premier est une procédure de remplissage et d'affichage d'une pile.
Code : C++
#include <iostream>
#include <stack>
using namespace std;
void remplir(stack<int>& p)
{
int n;
cout << "Entrez le nombre d'éléments de la pile : ";
cin >> n;
int elt;
for(int i = 0; i < n; i++)
{
cout << "Entrez un élément : ";
cin >> elt;
p.push(elt);
}
}
www.openclassrooms.com
Les piles et les files en C++ 7/16
int main()
{
stack<int> p;
remplir(p);
cout << "Votre pile est : ";
afficher(p);
return 0;
}
Code : C++
#include <iostream>
#include <stack>
using namespace std;
int main()
{
stack<int> p;
cout << "Apres declaration p = ";
afficher(p);
cout << "p.size() = " << p.size() << endl;
cout << "p.empty() = " << boolalpha << p.empty() << endl; /*
boolalpha permet d'afficher true ou false pour la valeur d'un
booléen (à la place de 1 ou 0) */
p.push(10);
p.push(20);
p.push(30);
cout << endl << "Apres empilement de 10, 20 et 30 p = ";
afficher(p);
cout << "p.size() = " << p.size() << endl;
cout << "p.empty() = " << boolalpha << p.empty() << endl;
cout << "p.top() = " << p.top() << endl;
p.pop();
cout << endl << "Apres p.pop(), p = ";
afficher(p);
www.openclassrooms.com
Les piles et les files en C++ 8/16
p.pop();
cout << endl << "Apres p.pop(), p = ";
afficher(p);
cout << "p.size() = " << p.size() << endl;
cout << "p.empty() = " << boolalpha << p.empty() << endl;
cout << "p.top() = " << p.top() << endl;
p.pop();
cout << endl << "Apres p.pop(), p = ";
afficher(p);
cout << "p.size() = " << p.size() << endl;
cout << "p.empty() = " << boolalpha << p.empty() << endl;
return 0;
}
Voilà, la partie sur les piles est terminée. Passons maintenant aux files .
Les files
Bien, les files maintenant !
Plusieurs de ces fonctions ont un fonctionnement identique à celles des piles, je les passerai donc plus rapidement.
Code : C++
#include <queue>
using namespace std;
Code : C++
www.openclassrooms.com
Les piles et les files en C++ 9/16
Pour les quatre fonctions qui vont suivre, je vais juste les citer pour la bonne et simple raison que j'ai déjà expliqué leur
fonctionnement dans la partie sur les piles.
La taille de la file
L'état de la file
Pour accéder au premier élément ajouté dans la pile (donc le premier à sortir), on utilise la fonction nomFile.front().
Pour accéder au dernier élément ajouté dans la liste, on utilise la fonction nomFile.back().
Code : C++
queue<int> i;
i.push(10);
i.push(20);
i.push(30);
i.push(40);
cout << "Le premier élément est : " << i.front() << endl;
cout << "Le dernier élément est : " << i.back() << endl;
i.pop();
cout << "Le premier élément est : " << i.front() << endl;
cout << "Le dernier élément est : " << i.back() << endl;
Le résultat est :
Code : Console
www.openclassrooms.com
Les piles et les files en C++ 10/16
Comme pour les piles, le passage de paramètres se fait de préférence par référence.
Code : C++
#include <iostream>
#include <queue>
using namespace std;
www.openclassrooms.com
Les piles et les files en C++ 11/16
while(!q.empty())
{
cout << q.front();
q.pop();
if(!q.empty())
cout << " , ";
}
}
Code : C++
#include <queue>
#include <iostream>
using namespace std;
int main()
{
queue<int> q;
q.push(10);
q.push(20);
q.push(30);
cout << endl << "Apres ajout de 10, 20 et 30, q = ";
while(!q.empty())
{
afficher(q);
cout << "q.size() = " << q.size() << endl;
cout << "q.front() = " << q.front() << endl;
cout << "q.back() = " << q.back() << endl;
q.pop();
cout << endl << "Apres q.pop(), q = ";
}
afficher(q);
return 0;
}
Voilà, la partie sur les files est terminée, et en plus, ça n'a même pas été dur !
Exemple d'utilisation
www.openclassrooms.com
Les piles et les files en C++ 12/16
Bien, nous allons maintenant voir un exemple concret avec les piles (un exemple avec les files suivra dès que je l'aurai trouvé !
).
Une expression est dite post-fixée ou écrite en notation polonaise inversée lorsque chaque opérateur est immédiatement précédé
de ses deux opérandes.
Les expressions suivantes en sont des exemples :
1. 42+
2. 32*54*+
3. 1253+*+7-
Pour les évaluer, on applique la définition : chaque opérateur est précédé immédiatement de ses deux opérandes.
Exemple 1 : 42+
On parcours l'epf jusqu'au premier opérateur qui est + et on l'applique aux deux opérandes qui le précèdent, 4 et 2, ce qui donne
au final 4+2=6.
Bon, c'était le plus simple !
Exemple 2 : 32*54*+
Lorsque l'expression devient plus compliquée, il est plus simple de faire un petit schéma pour comprendre. Pour faire ce schéma,
on écrit tous les chiffres sur la même ligne. Ensuite, on parcourt à nouveau l'epf jusqu'au premier opérateur et on l'applique aux
deux opérandes juste avant. Et on continue jusqu'à ce que tous les opérateurs aient été traités. On peut les relier avec des
flèches.
Pour cet exemple, on obtient ceci :
Exemple 3 : 1253+*+7-
Celui-ci est un peu plus complexe, mais en appliquant la méthode, on parvient au but très facilement.
Pour cet exemple, le schéma est :
www.openclassrooms.com
Les piles et les files en C++ 13/16
Comme vous pouvez le constater, si les epf deviennent très longues, c'est aussi très long à calculer. Nous allons donc faire le
programme qui les résout en nous aidant des piles.
Principe
À la fin, il ne reste plus qu'une seule valeur dans la pile, il s'agit du résultat de l'évaluation de l'epf.
Avec ces conditions, ce n'est pas trop dur d'écrire le programme. La seule difficulté qu'il peut y avoir est de faire la différence
entre un opérateur et une opérande. En effet, l'epf est donnée par l'utilisateur sous forme de string. Nous allons donc d'abord
coder deux fonctions : une pour chercher les opérateurs et l'autre pour chercher les opérandes.
Si vous avez regardé le tutoriel de M@teo21, vous avez sûrement pensé à la méthode itérative, c'est-à-dire avec une boucle. Je
vais vous montrer une autre méthode.
Voici la nouvelle méthode : dans la fonction, on déclare une chaîne contenant les opérateurs. Nous appliquerons la méthode
chaîne.find(char) sur cette chaîne. find prend un caractère en paramètre, et c'est elle qui va parcourir toute seule la chaîne pour
chercher le caractère passé en paramètre. Si le caractère est présent, elle renvoie la première position de celui-ci, sinon, elle
renvoie string::npos. Nous, nous allons utiliser le string::npos.
Voici le code de la fonction booléenne opérateur(const char c) qui renvoie vrai si le caractère c est un opérateur :
Code : C++
return res;
}
Ce caractère est celui lu dans l'epf. On lit les caractères de l'epf l'un après l'autre dans une procédure qui résout l'epf.
Passons maintenant à cette procédure. Elle est assez simple à écrire ; en effet, j'ai déjà donné la méthode plus haut ; il suffit de
l'appliquer bêtement !
Pour commencer, cette procédure prend une chaîne de caractères en paramètre. C'est aussi dans cette procédure que l'on va
initialiser la pile des opérandes et des différents résultats. Il nous faut également trois variables locales pour stocker la première
et la deuxième valeur de la pile et le résultat du calcul de ces deux valeurs.
www.openclassrooms.com
Les piles et les files en C++ 14/16
Code : C++
Normalement, il doit y avoir quelque chose qui vous choque dans ce code. J'ai mis :
Code : C++
p.push(epf.at(i)-'0');
C'est surtout le " -'0' " qui doit vous choquer. Je fais ça parce que epf.at(i) est un caractère et on ne peut pas utiliser le caractère
comme un chiffre, même si le caractère en question représente un chiffre.
Si on fait des calculs sur ce caractère, on fera les calculs avec les codes ASCII décimaux correspondants. Si on regarde une table
ASCII, ici par exemple, on voit que les codes ASCII décimaux des chiffres de 0 à 9 ont pour codes 48 pour 0... jusqu'à 57 pour 9.
On remarque alors qu'il suffit de retirer 48 à ce code pour avoir un code correspondant au chiffre. Pour retirer 48 au code ASCII, il
suffit de retirer le caractère 0, ce qui va retirer 48 au code ASCII du caractère auquel on applique cette soustraction.
Bien, comme je pense que vous avez bien souffert si vous lisez ceci, je vais vous donner le code complet du programme :
Code : C++
#include <iostream>
#include <stack>
#include <string>
www.openclassrooms.com
Les piles et les files en C++ 15/16
return res;
}
int main()
{
string epf;
return 0;
}
Citation : Xav57
Au revoir et à bientôt !
Partager
www.openclassrooms.com