Cours Java : les tableaux comment ca marche
Cours Java : les tableaux comment ça marche
…
Jusqu’ici, nous avons employé les variables pour stocker les valeurs individuelles de types primitifs : une variable de type int pour stocker un entier, une variable de type boolean pour un booléan, etc.
Un tableau est une structure regroupant plusieurs valeurs de même type, appelées composantes du tableau. On peut traiter un tableau comme un tout, ou composante par composante. Traité comme un tout, on pourra le stocker dans une variable, le passer en paramètre ou le donner en résultat d’un calcul. Chaque composante est désignée individuellement via son indice, qui correspond à sa position dans le tableau, et peut être traitée comme variable individuelle : on pourra consulter sa valeur, la modifier, etc.
↓
T[2] = composante d’indice 2
T[0], T[1], T[2], T[3], ..., T[6] } 7 Variables (composantes)
Les tableaux sont des structures des données présentes dans tous les langages de programmation. En général, chaque langage possède un type tableau prédéfini avec une syntaxe spécifique.
4.1 Déclaration et création
En Java, avant d’utiliser un tableau, il faut:
- Déclarer une variable de type tableau (symbole []), en indiquant le type T de ses futures composantes;
T [] tab; // tab est declare tableau de T
- Créer explicitement la structure du tableau en mémoire (opération new), en donnant sa taille et le type T de ses éléments. Cette taille ne pourra plus changer : en Java les tableaux sont de taille fixe.
tab = new T[taille];
- L’initialisation des composantes avec des valeurs par défaut, est réalisée implicitement par l’opération de création.
Étudions plus en détail ces étapes.
Déclaration L’instruction :
T [] tab;
déclare une variable tab destinée à contenir un tableau, dont les composantes seront de type T. Après déclaration, la variable tab existe, mais n’est pas encore initialisée à un tableau. En Java, on dira que tab ne référence pas encore 1 de tableau. Le dessin suivant montre l’état de tab en mémoire :
tab 6 −→
où le symbole 6−→ est lu : “n’est pas initialisé”. Exemples :
int [] tabNum; // tabNum est un tableau avec composantes de type int
double [] t; // t est un tableau avec composantes de type double
String [] m; // m est un tableau avec composantes de type String
tabNum[0] = 5; // provoque une erreur: le tableau n’existe pas
Après ces déclarations, les variables tabNum, t et m existent, mais pas encore la suite de composantes que chacune d’entre elles pourra désigner. Par exemple, il est impossible de modifier la première composante de tabNum (notée tabNum[0]) : elle n’existe pas encore. Le compilateur signale l’erreur :
Test.java:7: variable tabNum might not have been initialized tabNum[0] = 5;
Création
L’opération de création:
new T[n];
réalise la création et l’initialisation d’un tableau de n composantes de type T :
- Allocation en mémoire d’un espace suffisant pour stocker n composantes de type T.
- Initialisation des composantes du tableau avec des valeurs par défaut.
Les tableaux en Java sont de taille fixe. Une fois le tableau créé, l’espace qui lui est alloué en mémoire ne peut pas changer. Par exemple, il est impossible d’ajouter ou d’enlever des composantes d’un tableau.
Exemple 1 : Déclaration, puis création du tableau tab avec trois entiers.
- La notion de référence sera abordée dans le chapitre dédié aux objets.
2 NFA031 c~CNAM 2012
int [] tab; // Declaration
tab = new int[3]; // Creation
tab]0] = 7;
Après l’instruction de création new, la variable tab est initialisée (ou, fait référence) à un tableau contenant trois entiers. Après l’affectation tab[0] = 7, la structure du tableau en mémoire est:
tab −→ 7 0 0
Il est possible de réunir la déclaration et la création d’un tableau en une seule instruction. On pourra ainsi déclarer et créer le tableau de l’exemple 1 par:
Exemple 2 : Déclaration et création en une seule instruction.
int [] tab = new int[3];
Initialisation par une liste de valeurs
Lorsqu’un tableau est de petite taille, il est possible de l’initialiser en donnant la liste des valeurs de chaque composante. On utilise la notation {v0, v1, ... vn}, où vi est la valeur à donner à la composante i du tableau. Nous reprendrons souvent cette notation pour regrouper en une seule instruction la déclaration, la création et l’initialisation d’un tableau.
Exemple 3 : Déclaration, création et initialisation d’un tableau en une seule instruction.
int [] tab = {1,9,2,4};
Il est alors inutile de réaliser une création explicite via new : elle se fait automatiquement à la taille nécessaire pour stocker le nombre des valeurs données. En mémoire on aura:
tab −→ 1 9 2 4
Valeurs par défaut à la création
Lors de la création, les composantes d’un tableau sont initialisées avec des valeurs par défaut :
– les composantes boolean sont initialisées à false.
– les composantes numériques sont initialisées à 0.
– les composantes char sont initialisées au caractère nul ’\0’.
– les composantes référence 2 sont initialisées à la valeur null (référence nulle).
Exemple 4 : Valeurs par défaut dans les composantes après création.
int [] tb = new int[3];
char [] ch = new char[4];
boolean [] bt = new boolean[3];
- Voir le chapitre dédié aux objets.
L’opération new initialise les composantes de ces tableaux par:
tb −→ 0 0 0
ch −→ ’\0’ ’\0’ ’\0’ ’\0’
bt −→ false false false
Longueur d’un tableau
La taille ou longueur d’un tableau est le nombre n des composantes qu’il contient. Supposons que tab désigne un tableau de taille n. On peut obtenir sa longueur par la notation tab.lenght. Les indices du tableau tab sont alors compris entre 0 et tab.length-1. Cette notation sera souvent employée pour fixer la valeur maximale qui peut prendre l’indice d’un tableau (voir exemple 6).
4.2 Accès aux composantes
L’accès à une composante de tableau permet de traiter cette composante comme n’importe quelle variable individuelle: on peut modifier sa valeur dans le tableau, l’utiliser pour un calcul, un affichage, etc. L’accès d’une composante se fait via son indice ou position dans le tableau. En Java, la première position a pour indice 0, et la dernière, a l’indice n-1 (taille du tableau moins un).
premier indice → 0 i n-1 ← dernier indice (tab.length-1)
tab =
- Y ~
tab.length=n
tab[i] vaut 2.5
L’accès aux composantes de tab n’a de sens que pour les indices dans l’intervalle [0,..., tab.length-1]. Si i est un indice compris dans cet intervalle :
– tab[i] : est un accès à la composante de position i dans tab. On peut consulter ou modifier cette valeur dans le tableau. Exemple: tab[i] = tab[i] + 1;
– accès en dehors des bornes du tableau tab : si j n’est pas compris entre 0 et tab.length1, l’accès tab[j] provoque une erreur à l’exécution : l’indice j et la composante associée n’existent pas dans le tableau. Java lève l’exception ArrayIndexOutOfBoundsException.
Exemple 5 : Modification d’une composante, accès en dehors des bornes d’un tableau.
public class Test {
public static void main (String args[]) { double [] tab = {1.0, 2.5, 7.2, 0.6}; // Creation //Affichage avant modification Terminal.ecrireString("tab[0] avant = "); Terminal.ecrireDoubleln(tab[0]); tab[0] = tab[0] + 4;
//Affichage apr\‘es modification
Terminal.ecrireString("tab[0] apres = "); Terminal.ecrireDoubleln(tab[0]);
// tab[5] = 17; // Erreur: indice en dehors des bornes
}}
Ce programme affiche la valeur de la première composante de tab, avant et après modification.
Java/Essais> java Test tab[0] avant = 1.0 tab[0] apres = 5.0
Si nous enlevons le commentaire de la ligne 11, l’exécution se termine par une erreur: l’indice 5 est en dehors des bornes du tableau.
Java/Essais> java Test
tab[0] avant = 1.0
tab[0] apres = 5.0
Exception in thread "main" java.lang.ArrayIndexOutOfBoundsException: 5
at Test.main(Test.java:9)
Exemple 6 : Parcours pour affichage d’un tableau tab. La boucle fait varier l’indice de tab entre i= 0 et i <= tab.lenght-1.
public class AfficheTab {
public static void main (String args[]) {
int[] tab = {10,20,30,40};
for (int i=0; i<= tab.length -1; i++) {
Terminal.ecrireStringln("tab["+ i + "] = " + tab[i]);
}}}
Ce programma affiche:
Java/Essais> java AfficheTab
tab[0] = 10 tab[1] = 20 tab[2] = 30 tab[3] = 40
Une erreur commune dans une boucle, est de fixer le dernier indice à i <= tab.length plutôt qu’à i <= tab.length -1. Si nous changeons la condition de la boucle de cette manière, l’exécution produit une erreur: l’indice tab.length, égal à 4 ici, n’existe pas dans tab.
Java/Essais> java AfficheTabErr
tab[0] = 10
tab[1] = 20
tab[2] = 30
tab[3] = 40
Exception in thread "main" java.lang.ArrayIndexOutOfBoundsException: 4
at AfficheTabErr.main(AfficheTabErr.java:5)
Exemple 7 : Boucles d’initialisation et d’affichage. On souhaite initialiser un tableau avec des notes d’élèves saisies au clavier. Le programme demande le nombre de notes à saisir, et créé un tableau lesNotes de cette taille. Une première boucle initialise le tableau; la boucle suivante affiche son contenu. Les itérations se font dans l’intervalle de i=0 jusqu’à i <= lesNotes.length-1.
public class Notes {
public static void main (String args[]) {
int nombreNotes;
Terminal.ecrireString("Nombre de notes a saisir? ");
nombreNotes = Terminal.lireInt();
double [] lesNotes = new double[nombreNotes];
// Initialisation
for (int i=0; i<= lesNotes.length -1; i++) {
Terminal.ecrireString("Note no. " + (i+1) + "? ");
lesNotes[i] = Terminal.lireDouble();
}
// Affichage
Terminal.sautDeLigne();
Terminal.ecrireStringln("Notes dans le tableau:"); Terminal.ecrireStringln("**********************"); for (int i=0; i<= lesNotes.length -1; i++) {
Terminal.ecrireString("Note no. " + (i+1) + " = ");
Terminal.ecrireDoubleln(lesNotes[i]);
}}}
Java/Essais> java Notes
Nombre de notes a saisir? 4
Note no. 1? 7.6
Note no. 2? 11
Note no. 3? 14
Note no. 4? 5
Notes dans le tableau:
**** **** ** ** ****** ****
Note no. 1 = 7.6
Note no. 2 = 11.0
Note no. 3 = 14.0
Note no. 4 = 5.0
Affectation entre tableaux
Il est possible d’affecter une variable de type tableau à une autre autre variable, à condition qu’elles soient déclarées avec le même type de composantes. Après affectation, les deux variables référent à un même et seul tableau en mémoire: elles deviennent synonymes. Toute modification sur un composant de l’une modifie le même composant de l’autre.
Exemple 8 :
int [] t;
int [] m = {2,3,4,5,6};
t = m; // t et m designent un meme tableau Terminal.ecrireStringln("t[0] = " + t[0]); Terminal.ecrireStringln("m[0] = " + m[0]); //Modification de t[0]
t[0] = 9;
//Nouveaux t[0], m[0]
Terminal.ecrireStringln("Nouveau t[0] = " + t[0]); Terminal.ecrireStringln("Nouveau m[0] = " + m[0]);
L’affectation de m dans t a pour effet de bord de rendre ces deux variables synonymes : elles réfé-rent toutes les deux au même espace mémoire, qui contient le tableau {2, 3, 4, 5, 6} désigné par m. L’affectation t[0] = 9 est ainsi “visible” lors d’un accès à m. Ce programme affiche:
t[0] = 2
m[0] = 2
Nouveau t[0] = 9 Nouveau m[0] = 9
Enfin, les tableaux désignés par les variables d’une affectation peuvent avoir des longueurs différentes. Exemple 9 : Affectation entre tableaux de tailles différentes.
int [] t = {10, 20};
int [] m = {2,3,4,5,6};
Terminal.ecrireStringln("Longueur de t = " + t.length);
t = m; // t contient maintenant un tableau de 5 elements
//Nouvelle longueur de t
Terminal.ecrireStringln("Nouvelle longueur de t = " + t.length);
Ce programme affiche:
Longueur de t = 2
Nouvelle longueur de t = 5
Égalité entre tableaux
Lorsqu’on compare deux tableaux par des tests d’égalité (==) ou d’inégalité (!=), le test porte sur l’identité des tableaux et non sur leur contenu. Cela signifie qu’on cherche à savoir, non pas si les tableaux contiennent les mêmes éléments, mais s’ils ont été crées par un seul et même new. Voyons un exemple.
public class Tab5{
public static void main(String[] argv){
int[] tab1;
int[] tab2;
int[] tab3;
tab1 = new int[3];
tab1[0] = 10; tab1[1] = 20; tab1[2] = 30;
tab2 = new int[3]; tab2[0] = 10;
tab2[1] = 20;
tab2[2] = 30;
tab3 = tab2;
if (tab1 == tab2)1
Terminal.ecrireStringln("tab1 et tab2 sont egaux");
}else1
Terminal.ecrireStringln("tab1 et tab2 sont differents");
}
if (tab2 == tab3)1
Terminal.ecrireStringln("tab2 et tab3 sont egaux");
}else1
Terminal.ecrireStringln("tab2 et tab3 sont differents");
}}}
L’exécution du programme donne:
> java Tab5
tab1 et tab2 sont différents
tab2 et tab3 sont égaux
Explication: tab1 et tab2 sont deux tableaux dont les contenus sont égaux, mais ces deux tableaux ont été créés par deux new différents. Il s’agit donc de deux espaces mémoires distincts. Changer l’un ne change pas l’autre. En revanche, tab2 et tab3 sont créés par un seul et unique new. Changer le contenu de l’un change aussi le contenu de l’autre. Il s’agit d’un seul et unique espace mémoire.
4.3 Exemples avec tableaux unidimensionnels
Exemple 10 : Recherche du minimum et maximum dans un tableau d’entiers. Deux variables min
et max sont initialisées avec le premier élément du tableau. La boucle de recherche des minimum et maximum, compare chaque élément avec ces deux valeurs. La comparaison se fait à partir du deuxième élément: c’est pourquoi l’indice i débute à i=1.
public class minMax 1
public static void main (String args[]) 1
int n;
Terminal.ecrireString("Combien des nombres a‘ saisir? ");
n = Terminal.lireInt();
int [] tab = new int[n];
// Initialisation
for (int i=0; i<= tab.length -1; i++) 1
Terminal.ecrireString("Un entier? ");
tab[i] = Terminal.lireInt();
}
// Recherche de min et max
int min = tab[0];
int max = tab[0];
for (int i=1; i<= tab.length -1; i++) 1
if (tab[i] < min) 1 min= tab[i];}
if (tab[i] > max) 1 max = tab[i];}
}
Terminal.ecrireStringln("Le minimum est: " + min); Terminal.ecrireStringln("Le maximum est: " + max);
}
}
Voici une exécution du programme:
Java/Essais> java minMax
Combien des nombres a‘ saisir? 5
Un entier? 5 Un entier? 2 Un entier? -6 Un entier? 45 Un entier? 3 Le minimum est: -6
Le maximum est: 45
Exemple 11 : Gestion de notes. Nous modifions le programme de l’exemple 7 afin de calculer la
moyenne des notes, la note minimale et maximale, et le nombre de notes supérieures ou égales à 10. Nous reprenons la boucle de recherche du minimum et maximum de l’exemple 10.
public class Notes {
public static void main (String args[]) {
int nombreNotes;
Terminal.ecrireString("Nombre de notes a saisir? ");
nombreNotes = Terminal.lireInt();
double [] lesNotes = new double[nombreNotes];
// Initialisation
for (int i=0; i<= lesNotes.length -1; i++) {
Terminal.ecrireString("Note no. " + (i+1) + "? ");
lesNotes[i] = Terminal.lireDouble();
}
double min = lesNotes[0];
double max = lesNotes[0];
double somme = 0;
int sup10 = 0;
for (int i=0; i<= lesNotes.length -1; i++) {
if (lesNotes[i] < min) { min = lesNotes[i];}
if (lesNotes[i] > max) { max = lesNotes[i];}
if (lesNotes[i] >= 10) { sup10++;}
somme = somme + lesNotes[i];
}
Terminal.ecrireString("La moyenne des notes est: "); Terminal.ecrireDoubleln(somme/nombreNotes); Terminal.ecrireStringln("Le nombre de notes >= 10 est: " + sup10); Terminal.ecrireStringln("La note minimum est: " + min); Terminal.ecrireStringln("La note maximum est: " + max);
}}
Java/Essais> java Notes Nombre de notes a‘ saisir? 4
Note no. 1? 5
Note no. 2? 8
Note no. 3? 10
Note no. 4? 15
La moyenne des notes est: 9.5
Le nombre de notes >= 10 est: 2
La note minimum est: 5.0
La note maximum est: 15.0
Exemple 12 : Inversion d’un tableau de caractères. La boucle d’inversion utilise deux variables d’itération i et j, initialisées avec le premier et le dernier élément du tableau. A chaque tour de boucle, les éléments dans i et j sont échangés, puis i est incrémenté et j décrémenté. Il y a deux cas d’arrêt possibles selon la taille du tableau : s’il est de taille impair, alors l’arrêt se produit lorsque i=j ; s’il est de taille pair, alors l’arrêt se fait lorsque j < i. En conclusion, la boucle doit terminer si i >= j.
public class Inversion {
public static void main (String[] args) {
int n;
char [] t;
Terminal.ecrireString("Combien de caracteres a saisir? ");
n = Terminal.lireInt();
t = new char[n];
// Initialisation
for (int i=0; i<= t.length-1; i++) {
Terminal.ecrireString("Un caractere: ");
t[i] = Terminal.lireChar();
}
// Affichage avant inversion
Terminal.ecrireString("Le tableau saisi: "); for (int i=0; i<= t.length-1; i++) { Terminal.ecrireChar(t[i]);
}
Terminal.sautDeLigne();
// Inversion: arret si (i >= j)
int i,j;
char tampon;
for (i=0, j= t.length-1 ; i < j; i++, j--) {
tampon = t[i];
t[i] = t[j];
t[j] = tampon;
}
// Affichage final
Terminal.ecrireString("Le tableau inverse: "); for (int k=0; k<= t.length-1; k++) { Terminal.ecrireChar(t[k]);
}
Terminal.sautDeLigne();
}
}
Un exemple d’exécution :
10 NFA031 c~CNAM 2012
Java/Essais> java Inversion
Combien de caracteres a saisir? 5
Un caractere: s Un caractere: a Un caractere: l Un caractere: u Un caractere: t Le tableau saisit: salut
Le tableau inverse: tulas
4.4 Le tri par sélection
Les programmes de tri de tableaux sont couramment utilisés dans la gestion des données. Nous présentons un algorithme simple de tri, le tri par sélection. Il est appelé ainsi car, lors de chaque itération, il sélectionne l’élément le plus petit parmi ceux restant à trier et le met à sa place dans le tableau, en l’échangeant avec l’élément qui s’y trouve. Nous illustrons cette méthode sur le tableau {4, 5, 1, 9, 8}.
La première fois, l’algorithme sélectionne le plus petit du tableau, qui est 1, et l’échange avec l’élément qui se trouve à la première place (4) :
{4, 5, 1 ,9,8}
après échange { 1 ,5,4,9,8}
Au bout de cette première itération, le premier élément est trié : c’est le plus petit de tout le tableau. Les éléments restant à trier sont ceux à partir de la 2ème position. La fois suivante, le plus petit parmi eux (4), est sélectionné et échangé avec l’élément se trouvant à la deuxième place (5) :
{1, 5, 4 ,9,8}
après échange {1, 4 ,5,9,8}
Après la deuxième itération, les deux premiers éléments sont triés : la première place contient le plus petit, la deuxième, contient le deuxième plus petit. L’algorithme finit lorsqu’il ne reste plus qu’un seul seul élément à trier, qui se trouve nécessairement à sa place: la dernière. Terminons l’application de l’algorithme :
{1, 4, 5 ,9,8} : 5 est échangé avec lui-même
{1, 4, 5, 9, 8 } : 8 est échangé avec 9
{1, 4, 5, 8, 9} : Le tableau est trié
Algorithme de tri par sélection :
Entrée : un tableau tab taille n Sortie : le tableau tab trié.
Pour chaque indice i de tab compris dans [0, ..., n-2], faire :
- Sélectionner le plus petit élément parmi ceux d’indices [i, ..., n-1], et déterminer son indice Im.
- Échanger les éléments tab[Im] et tab[i].
Le pas de sélection du plus petit élément se fait également avec une boucle. La technique employée est similaire à celle de l’exemple 10, mais ici, la recherche ne se fait pas sur tout le tableau, mais seulement sur la partie restant à trier lors de chaque itération.
Exemple 13 : Tri par sélection.
public class triSelection {
public static void main (String args[]) {
int n;
int [] tab;
Terminal.ecrireString("Nombre de entiers a trier? ");
n = Terminal.lireInt();
tab = new int[n];
//Initialisation de tab
for (int i = 0; i <= tab.length -1; i++) {
Terminal.ecrireString("Un entier? ");
tab[i] = Terminal.lireInt();
}
// Tri
for (int i = 0; i <= tab.length-2; i++) {
//Recherche du min dans [i .. tab.lentgh-1]
int Im = i;
for (int j = i+1 ; j <= tab.length-1; j++) {
if (tab[j] < tab[Im]) {Im = j;}
}
//Echange de tab[i] avec le min trouve
int tampon = tab[i];
tab[i] = tab[Im];
tab[Im] = tampon;
}
//Affichages
Terminal.ecrireString("Tableau trie: ");
for (int i = 0; i <= tab.length -1; i++) {
Terminal.ecrireString(" ");
Terminal.ecrireInt(tab[i]);
}
Terminal.sautDeLigne();
}
}
Voici un exemple d’exécution :
Java/Essais> java triSelection Nombre d’entiers a trier? 6
Un entier? 10 Un entier? 2 Un entier? 7 Un entier? 21 Un entier? 8 Un entier? 5 Tableau trie: 2 5 7 8 10 21
Nous présentons une trace partielle de ce programme, sur le tableau tab = {4,5,1,9,8 }. Elle montre, pour chaque pas de boucle, la position de l’indice i, et celle de l’indice Im du plus petit élément parmi ceux restant à trier :
Itération 1 : i = 0, Im = 2.
i↓ Im↓
1
Après échange:
Après échange:
1 4 5 9 8
Itération 3 : i = 2, Im = 2.
Im i↓
Après échange: rien ne change.
Itération 4 : i = 3, Im = 4.
i↓ Im↓
Après échange, il ne reste plus qu’un élément: le tableau est donc trié.
1 4 5 8 9
4.5 Tableaux à deux dimensions
Les tableaux vus jusqu’ici sont des tableaux à une dimension: conceptuellement tous les éléments se trouvent dans une seule ligne (ou colonne, cela revient au même). Les tableaux à plusieurs dimensions sont utiles dans la modélisation des données, mais ce sont les tableaux à deux dimensions qui sont de loin les plus utilisés en informatique. Nous concentrons notre étude à leur cas.
Un tableau à deux dimensions, ou matrice, représente un rectangle composé de lignes et de colonnes. Chaque élément stockée dans le tableau est adressé par sa position, donnée par sa ligne et sa colonne. En Java, si tab est un tableau à deux dimensions, l’élément de ligne i et colonne j est
désigné par tab[i][j].
colonnes
Déclaration
En Java, un tableau de n dimensions et composantes de type T est déclaré par:
T [] [] $\ldots$ [] tab; // n fois le symbole []
Chaque ajout du symbole [] permet d’obtenir une dimension supplémentaire:
[] t; // une dimension
[] [] m; // deux dimensions
[] [] [] p; // trois dimensions
Création
Comme avant, nous utilisons l’opération new de création de tableaux, en donnant en plus du type des éléments, la taille de chaque dimension.
Exemple 14 : Création d’un tableau d’entiers à deux dimensions, avec 3 lignes et 5 colonnes. Après création, nous modifions l’élément de la ligne 1 et colonne 2 par t[1][2] = 7.
int [][] t = new int [3][5]; // 3 lignes, 5 colonnes
t[1][2] = 7;
Nous pouvons imaginer t comme un rectangle avec 3 lignes et 5 colonnes. Par convention, la pre-mière dimension est celle des lignes, et la deuxième, celle des colonnes. Comme avant, les indices débutent à 0. Lors de sa création, les éléments du tableau sont initialisés à 0 par défaut.