Programmation C++/Les structures de contrôles
Une série d'instructions dans une fonction s'exécute séquentiellement par défaut. Cependant, il est nécessaire que certaines parties du code ne s'exécutent que sous certaines conditions, ou s'exécutent plusieurs fois dans une boucle pour par exemple traiter tous les éléments d'un tableau.
Le if
modifierCette structure de contrôle permet d'exécuter une instruction ou une suite d'instructions seulement si une condition est vraie.
Syntaxe :
modifierif (condition) instruction
- condition
- Expression booléenne de la condition d'exécution.
- instruction
- Une instruction ou un bloc d'instructions entre accolades exécuté si la condition est vraie.
Lorsqu'il n'y a qu'une ligne d'instructions conditionnelles, la paire d'accolades est conditionnelle, mais les omettre est dangereux.
Sémantique :
modifierOn évalue la condition :
- si elle est vraie on exécute l’instruction et on passe à l’instruction suivante.
- si elle est fausse on passe directement à l’instruction suivante.
L’instruction peut être remplacée par une suite d’instructions entre accolades.
Exemple :
modifier#include <iostream>
using namespace std;
int main()
{
int a;
cout << "Tapez la valeur de a : "; cin >> a;
if (a > 10)
{
cout << "a est plus grand que 10" << endl;
}
cout << "Le programme est fini." << endl;
return 0;
}
Ce programme demande à l'utilisateur de taper un entier a.
Si a est strictement plus grand que 10, il affiche "a est plus grand que 10."
.
Dans le cas contraire, il n'affiche rien.
Exécution 1
modifierTapez la valeur de a : 12 a est plus grand que 10
Exécution 2
modifierTapez la valeur de a : 8
Condition multiple contenant des opérateurs logiques
modifierSi la condition a évaluer est complexe et contient des opérateurs logiques, chaque condition doit être écrite entre parenthèse.
Par exemple, si la condition condition1 et la condition condition2 doivent être vérifiée en même temps, on écrira
if ( (condition1) && (condition2) ) instruction
où l'opérateur logique && peut être remplacé par tout autre opérateur logique.
Ajouter des opérateurs logiques se fait selon le même principe, en prenant en compte les règles de préséance de ces opérateurs pour adapter la place des parenthèses.
Le if...else
modifierCette structure de contrôle permet d'exécuter une instruction si une condition est vraie, ou une autre instruction si elle est fausse.
Syntaxe
modifierif (condition) instruction1 else instruction2
- condition
- Expression booléenne de la condition d'exécution.
- instruction1
- Une instruction ou un bloc d'instructions exécuté si la condition est vraie.
- instruction2
- Une instruction ou un bloc d'instructions exécuté si la condition est fausse.
Sémantique
modifierOn évalue la condition :
- si elle est vraie, on exécute l'
instruction1
et on passe à l'instruction suivante - si elle est fausse, on exécute l'
instruction2
et on passe à l'instruction suivante
L'instruction1
ou l'instruction2
peuvent être remplacées par une suite d'instructions entre accolades.
Lorsqu'il n'y a qu'une ligne d'instructions conditionnelles, la paire d'accolades est conditionnelle, mais les omettre est dangereux.
Exemple :
modifier#include <iostream>
using namespace std;
int main()
{
int a;
cout << "Tapez la valeur de a : "; cin >> a;
if (a > 10)
{
cout << "a est plus grand que 10" << endl;
}
else
{
cout << "a est inférieur ou égal à 10" << endl;
}
return 0;
}
Ce programme demande à l'utilisateur de taper un entier a
. Si a est strictement plus grand que 10, il affiche "a est plus grand que 10"
. Dans le cas contraire, il affiche "a est inférieur ou égal à 10"
.
Exécution 1
modifierTapez la valeur de a : 12 a est plus grand que 10
Exécution 2
modifierTapez la valeur de a : 8 a est inférieur ou égal à 10
Plusieurs instructions par condition
modifierSi plusieurs actions doivent s'enchaîner lorsqu'une condition est vraie, on met obligatoirement ces actions entre accolades. L'accolade fermante ne doit pas être suivie d'un point virgule dans ce cas. Ainsi, on écrira
if (condition1) { instruction1.1 ; instruction1.2 ; ... } // pas de point virgule ici else if (condition2) { instruction2.1 ; ... } ....
qui exécute les instructions instruction1.1 et instruction1.2, ... si la condition condition1 est vraie, et les instructions instruction2.1, ... si la condition condition2 est vraie. Lorsqu'il n'y a qu'une ligne d'instructions conditionnelles, la paire d'accolades est conditionnelle, mais les omettre est dangereux.
Le switch
modifierL'instruction switch
permet de tester plusieurs valeurs pour une expression.
Syntaxe
modifierswitch(expression) { case constante1: instruction1_1 instruction1_2... case constante2: instruction2_1 instruction2_2... ... default: instruction_1 instruction_2... }
- expression
- Expression de type scalaire (entier, caractère, énumération, booléen).
- case constante1: instruction1_...
- Une série d'instructions ou de blocs d'instructions exécutés si expression vaut constante1.
- case constante2: instruction2_...
- Une série d'instructions ou de blocs d'instructions exécutés si expression vaut constante2.
- default: instruction_...
- Une série d'instructions ou de blocs d'instructions exécutés quand aucun des cas précédent ne correspond.
Sémantique
modifierOn teste la valeur de l'expression spécifiée.
On la compare successivement aux constantes constante1
, constante2
, etc.
Si la valeur de l'expression correspond à l'une des constantes, l'exécution débute à l'instruction correspondante.
Si aucune constante ne correspond, l'exécution débute à l'instruction correspondant à default:
.
L'exécution se termine à l'accolade fermante du switch
ou avant si l'instruction break;
est utilisée.
En général, l'instruction break
sépare les différents cas. Il n'est pas utilisé quand plusieurs cas sont traités par les mêmes instructions.
En C++, l'expression n'est jamais une chaine de caractères, ni un tableau, ni un objet, ni une structure.
Exemple 1
modifier#include <iostream>
using namespace std;
int main()
{
int a;
cout << "Tapez la valeur de a : "; cin >> a;
switch(a)
{
case 1 :
cout << "a vaut 1" << endl;
break;
case 2 :
cout << "a vaut 2" << endl;
break;
case 3 :
cout << "a vaut 3" << endl;
break;
default :
cout << "a ne vaut ni 1, ni 2, ni 3" << endl;
break;
}
return 0;
}
Ce programme demande à l'utilisateur de taper une valeur entière et la stocke dans la variable a. On teste ensuite la valeur de a : en fonction de cette valeur on affiche respectivement les messages "a vaut 1"
, "a vaut 2"
, "a vaut 3"
, ou "a ne vaut ni 1, ni 2, ni 3"
.
Exécution 1 :
Tapez la valeur de a : 1 a vaut 1
Exécution 2 :
Tapez la valeur de a : 2 a vaut 2
Exécution 3 :
Tapez la valeur de a : 3 a vaut 3
Exécution 4 :
Tapez la valeur de a : 11 a vaut ne vaut ni 1, ni 2, ni 3
Exemple 2
modifier#include <iostream>
using namespace std;
int main()
{
int a;
cout << "Tapez la valeur de a : "; cin >> a;
switch(a)
{
case 1 :
cout << "a vaut 1" << endl;
break;
case 2 :
case 4 :
cout << "a vaut 2 ou 4" << endl;
break;
case 3 :
case 7 :
case 8 :
cout << "a vaut 3, 7 ou 8" << endl;
break;
default :
cout << "valeur autre" << endl;
break;
}
return 0;
}
Ce programme demande à l'utilisateur de taper une valeur entière et la stocke dans la variable a. On teste ensuite la valeur de a : en fonction de cette valeur on affiche respectivement les messages "a vaut 1"
, "a vaut 2 ou 4"
, "a vaut 3, 7 ou 8"
, ou "valeur autre"
.
Exécution 1 :
Tapez la valeur de a : 1 a vaut 1
Exécution 2 :
Tapez la valeur de a : 4 a vaut 2 ou 4
Exécution 3 :
Tapez la valeur de a : 7 a vaut 3, 7 ou 8
Exécution 4 :
Tapez la valeur de a : 11 valeur autre
L'instruction switch
de cet exemple regroupe le traitement des valeurs 2 et 4, et des valeurs 3, 7 et 8.
Le for "classique"
modifierLe for est une structure de contrôle qui permet de répéter un certain nombre de fois une partie d'un programme.
Syntaxe
modifierfor(instruction_init ; condition ; instruction_suivant) instruction_répétée
- instruction_init
- Une instruction (ou une série d'instruction séparées par une virgule) d'initialisation de la boucle (initialiser un compteur à 0, pointer le premier élément d'une liste, ...).
- condition
- Expression booléenne de la condition de répétition de la boucle.
- instruction_suivant
- Une instruction (ou une série d'instruction séparées par une virgule) pour passer à l'itération suivante (incrémenter un index, passer à l'élément suivant, ...)
- instruction_répétée
- Une instruction ou un bloc d'instruction répété à chaque itération de la boucle.
Sémantique
modifier- on exécute l'
instruction_init
. - On teste la
condition
:- tant qu'elle est vraie, on exécute l'
instruction_repetée
, puis l'instruction_suivant
puis on revient au 2. - si elle est fausse, la boucle est terminée et on passe à l'instruction suivante.
- tant qu'elle est vraie, on exécute l'
- L'
instruction_repetée
peut être une suite d'instructions entre accolades.
Exemples
modifierPremier cas simple :
#include <iostream>
using namespace std;
int main()
{
int i;
for(i=0 ; i<10 ; i++)
cout << "BONJOUR" << endl;
return 0;
}
Second cas où deux index dans un tableau sont utilisés pour le parcourir à partir des deux bouts : le début et la fin.
int main()
{
// 10 entiers indexés de 0 à 9 inclus :
int elements[10] = { 3, 14, 15, 9, 26, 5, 35, 8 };
int i, j;
for (i=0, j=9; i <= j; i++, j--)
cout << "Elements"
<< " [" << i << "]=" << elements[i]
<< " [" << j << "]=" << elements[j]
<< endl;
return 0;
}
Instructions multiples
modifierIl existe également une pratique répandue qui consiste à initialiser plusieurs variables, évaluer plusieurs conditions ou, plus fréquemment, modifier plusieurs variables. Exemple :
float c=10;
for(int a=2, b=a; a<2, b*=a; a++, c/=b)
{
}
Ceci compilera parfaitement, mais est également parfaitement illisible. Dans la plupart des cas, il vaut mieux n'utiliser la boucle for elle-même que pour traiter le ou les compteurs, afin de conserver un code compréhensible. Cela compile parce qu'en C++, la virgule ',' est un opérateur comme les autres, qui évalue l'expression qui la précède et celle qui la suit, et renvoie le résultat de la dernière. Pour donner un exemple plus clair :
int b = 2,10; // b=10. Vous pouvez compiler pour vérifier.
Boucle infinie
modifierQuand aucune condition n'est spécifiée, celle-ci vaut true
par défaut, ce qui créé donc une boucle infinie.
Une boucle infinie est parfois nécessaire quand on ne connait pas l'instant où la boucle doit être arrêtée, et que la condition d'arrêt est évaluée au cœur de la boucle (gestion de messages, écoute de connexions réseau, ...).
for(;;)
{
// ...
}
le for "moderne"
modifierLe dernier standard C++ (surnommé C++11) a ajouté une nouvelle syntaxe pour la boucle for. En effet, on s'est aperçu que l'on utilise le plus souvent la boucle for pour parcourir une collection (une collection est un tableau classique ou un conteneur comme ceux de la stl.) en entier, et que donc le même code revenait très souvent (à la langue des commentaires près ;) ). Voici des exemples de code fréquents :
typedef std::list<int> MaListe;
MaListe maListe;
for(MaListe::iterator it=maListe.begin();it!=maListe.end();++it)
{
}
int tableau[]={0,1,2,3,4};
const int tableauMax=5;//il y a 5 cases dans tableau
for(int i=0; i<tableauMax; ++i)
{
}
Très répétitif, non (surtout le 1er exemple d'ailleurs) ? Donc, une nouvelle syntaxe a été créée qui permet de s'affranchir d'une grande partie de ces frappes et donc de prolonger la durée de vie de nos claviers ;)
Syntaxe
modifierfor(variable : collection) instruction
Sémantique
modifiervariable indique la variable qui recevra les valeurs issues de collection, en commençant par la première jusqu'à la dernière. Si variable n'existe pas, il est possible de la déclarer dans la même ligne, de la même façon que pour l'écriture classique.
Un bémol de cette syntaxe est que le compteur est perdu : si vous avez besoin du compteur dans le bloc d'instruction (ou plus tard) cette syntaxe est inutilisable.
Vous pouvez néanmoins, créer un compteur en le déclarant avant la boucle (en fixant sa valeur de départ à 0, par exemple) et l'entretenir durant l'exécution de la boucle (en l'incrémentant, par exemple).
Exemple
modifierAppliqué au dernier exemple, voici le résultat :
typedef std::list<int> MaListe;
MaListe maListe;
for(int i : maListe)
{
}
int tableau[]={0,1,2,3,4};
for(int i: tableau)
{
}
//pour garder le compteur
int tableau[]={0,1,2,3,4};
int cpt = 0;
for(int i: tableau)
{
++cpt;
}
Beaucoup plus clair, non ?
Le while
modifierSyntaxe
modifierwhile (condition) instruction;
Sémantique
modifierOn teste la condition :
- si elle est vraie, on exécute l'instruction et on recommence.
- si elle est fausse, la boucle est terminée, on passe à l'instruction suivante.
L'instruction peut être une suite d'instructions entre accolades.
Exemple de programme
modifier#include <iostream>
using namespace std;
int main()
{
int i = 0;
while (i < 10)
{
cout << "La valeur de i est : " << i << endl;
i++;
}
cout << "La valeur finale de i est : " << i << endl;
return 0;
}
Exécution
modifierLa valeur de i est : 0 La valeur de i est : 1 La valeur de i est : 2 La valeur de i est : 3 La valeur de i est : 4 La valeur de i est : 5 La valeur de i est : 6 La valeur de i est : 7 La valeur de i est : 8 La valeur de i est : 9 La valeur finale de i est : 10
Explications
modifierLa variable i est initialisée à 0.
À chaque étape, à la fin du corps du while, on incrémente i de 1.
On exécute donc le corps du while la première fois avec i valant 0, la dernière fois avec i valant 9.
Lorsqu’on sort du while, i vaut 10.
Le do ... while
modifierSyntaxe
modifierdo { ...instructions... } while( condition );
Sémantique
modifier- on exécute les instructions ;
- on évalue la condition ;
- si elle est vraie, on recommence au 1 ;
- si elle est fausse, la boucle est terminée, on passe à l ’instruction suivante.
Exemple
modifier#include <iostream>
using namespace std;
int main()
{
int i = 0;
do
{
cout << "La valeur de i vaut : " << i << endl;
i++;
}
while (i < 10);
cout << "La valeur finale de i est " << i << endl;
return 0;
}
Exécution :
La valeur de i vaut : 0 La valeur de i vaut : 1 La valeur de i vaut : 2 La valeur de i vaut : 3 La valeur de i vaut : 4 La valeur de i vaut : 5 La valeur de i vaut : 6 La valeur de i vaut : 7 La valeur de i vaut : 8 La valeur de i vaut : 9 La valeur finale de i est : 10
Le goto
modifierLe goto
est une structure de contrôle obsolète et ne doit plus être utilisé.
Le break
modifierL'instruction break
sert à "casser" ou interrompre une boucle (for
, while
et do
), ou un switch
.
Syntaxe
modifierbreak;
Sémantique
modifier- Sort de la boucle ou de l'instruction
switch
la plus imbriquée.
Exemple
modifier#include <iostream>
using namespace std;
int main()
{
for (int i = 0; i < 10; i++)
{
cout << "La variable i vaut : " << i << endl;
if (i == 5)
break;
}
return 0;
}
Ce programme interrompt sa boucle lors de la sixième itération.
Exécution
modifierLa variable i vaut : 0 La variable i vaut : 1 La variable i vaut : 2 La variable i vaut : 3 La variable i vaut : 4 La variable i vaut : 5
Note importante
modifierL'instruction break
interrompt le cours normal de l'exécution et doit être utilisée avec parcimonie.
Elle ne doit servir que pour éviter une complexification de la condition ; une condition complexe pouvant entraîner des erreurs de code et une moindre lisibilité.
L'exemple ci-dessous montre un cas où il n'est pas pertinent d'utiliser l'instruction break
.
for (int i=0 ; i<10 ; i++)
{
cout << "La variable i vaut : " << i << endl;
if (i == 5) break;
}
L'arrêt par l'instruction est notamment non pertinent dans ce cas car l'arrêt est prévisible à l'avance. Il suffit pour cet exemple de corriger la condition d'arrêt de la boucle :
for (int i=0 ; i<=5 ; i++)
{
cout << "La variable i vaut : " << i << endl;
}
Par contre, dans le cas de l'exemple suivant, où un arrêt prématuré de la boucle est effectué lorsqu'un item de tableau est trouvé et qu'un traitement complexe détermine qu'il faut interrompre la boucle, l'utilisation de l'instruction break
est pertinent.
L'équivalent avec une condition de boucle serait complexe, ou nécessiterait l'utilisation d'une variable dédiée au stockage de la condition.
for (int i=0 ; i<10 ; i++)
{
cout << "ITEM " << i << endl;
if (item[i]->getType() == type_a_trouver)
{
cout << " Trouvé à l'index " << i << endl;
if (item[i]->analyseContenuFichier() == TYPE_WIKICODE)
{
cout << " Page wiki trouvée." << endl;
break;
}
cout << " Fausse alerte, on continue..." << endl;
}
cout << " Item suivant !" << endl;
}
L'équivalent avec une variable stockant l'état de la condition de sortie impliquerait de tester cet état dans la condition de boucle, déterminer une valeur initiale pertinente pour que la boucle fasse la première itération, de tester cet état aussi avant chaque instruction ou bloc à chaque niveau d'accolade. Cela produit un code beaucoup moins explicite, et réduit la maintenabilité du code (répétition de code) :
int trouve = 0;
for (int i=0 ; i<10 && trouve==0 ; i++)
{
cout << "ITEM " << i << endl;
if (item[i]->getType() == type_a_trouver)
{
cout << " Trouvé à l'index " << i << endl;
if (item[i]->analyseContenuFichier() == TYPE_WIKICODE)
{
cout << " Page wiki trouvée." << endl;
trouve = 1;
}
if (trouve == 0)
cout << " Fausse alerte, on continue..." << endl;
}
if (trouve == 0)
cout << " Item suivant !" << endl;
}
Le continue
modifierL'instruction continue sert à "continuer" une boucle (for
, while
et do
) avec la prochaine itération.
Syntaxe
modifiercontinue;
Sémantique
modifierPasse à l'itération suivante de la boucle la plus imbriquée.
Exemple
modifier#include <iostream>
using namespace std;
int main()
{
for (int i = 0; i < 10; i++)
{
if (i == 5) continue;
cout << "La variable i vaut : " << i << endl;
}
return 0;
}
Ce programme saute l'affichage de la sixième itération.
Exécution
modifierLa variable i vaut : 0 La variable i vaut : 1 La variable i vaut : 2 La variable i vaut : 3 La variable i vaut : 4 La variable i vaut : 6 La variable i vaut : 7 La variable i vaut : 8 La variable i vaut : 9