Cet exercice ne nécessite pas de corrigé
Définissez une classe Vehicule qui a pour attributs des informations valables pour tout type de véhicule : sa marque , sa date d'achat, son prix d'achat et son prix courant.
Solution :
class Vehicule { private String marque; private int dateAchat; private double prixAchat; private double prixCourant; |
Ces attributs sont private pour assurer une bonne encapsulation.
Définissez un constructeur prenant en paramètres la marque, la date d'achat et le prix d'achat.
Une solution possible: :
public Vehicule(String marque, int date, double prix) { this.marque = marque; dateAchat = date; prixAchat = prix; prixCourant = prix; } |
Définissez une méthode publique affiche() qui affiche la valeur des attributs.
public void affiche() { System.out.print("marque : " + marque + ", date d'achat : " + dateAchat + ", prix actuel : " + prixCourant); System.out.println(); } |
Définissez deux classes Voiture et Avion, héritant de la classe Vehicule et ayant les attributs supplémentaires suivants :
Commençons par la classe Voiture. Elle doit hériter de la classe Vehicule :
class Voiture extends Vehicule { |
On ajoute ensuite les champs spécifiques à la classe Voiture :
private double cylindree; private int nbPortes; private double puissance; private double kilometrage; |
Pour la classe Avion, on procède de même :
class Avion extends Vehicule { private String moteur; private int heuresVol; |
Définissez maintenant un constructeur pour Voiture, ainsi qu'une méthode affichant la valeur des attributs.
public Voiture(String marque, int date, double prix, double cylindree, int portes, double cv, double km) { super(marque, date, prix); this.cylindree = cylindree; nbPortes = portes; puissance = cv; kilometrage = km; } |
public void affiche() { System.out.println(" ---- Voiture ----"); super.affiche(); System.out.println(cylindree + " litres, " + nbPortes + " portes, " + puissance + " CV, " + kilometrage + " km."); } |
Ces deux méthodes doivent, bien entendu, être publiques puisqu'elles sont précisément faites pour être utilisées hors de la classe.
Notez que pour le constructeur de Voiture, on fait appel au constructeur de Vehicule . On fait également appel à la méthode d'affichage de la super-classe dans la méthode affiche de Voiture.
Les méthodes de la classe Avion s'implémentent de même :
public Avion(String marque, int date, double prix, String moteur, int heures) { super(marque, date, prix); this.moteur = moteur; heuresVol = heures; } |
public void affiche() { System.out.println(" ---- Avion à " + moteur + "----"); super.affiche(); System.out.println(heuresVol + " heures de vol."); } |
Ajoutez une méthode void calculePrix() dans la
classe Vehicule
Le prix doit rester positif (donc s'il est négatif, on le met à 0).
public void calculePrix(int anneActuelle) { double decote = (anneActuelle - dateAchat) * 0.01; prixCourant = Math.max(0.0, (1.0 - decote) * prixAchat); } |
Redéfinissez cette méthode dans les deux sous-classes Voiture et Avion.
Les entêtes sont les mêmes pour Voiture et Avion que pour Vehicule, par contre les définitions diffèrent.
Voiture:
public void calculePrix(int anneActuelle) { double decote = (anneActuelle - getDateAchat()) * .02; // On force le type en int de manière à arrondir le résultat // On verra dans quelques semaines une manière plus élégante de faire // ce genre de choses... decote += 0.05 * (int)(kilometrage / 10000); if (getMarque().equals("Fiat") || getMarque().equals("Renault")) { decote += 0.1; } else if (getMarque().equals("Ferrari") || getMarque().equals("Porsche")) { decote -= 0.2; } setPrixCourant(Math.max(0.0, (1.0 - decote) * getPrixAchat())); } |
On se rend compte ici que l'on a besoin d'accèder à des attributs privés de la super-classe, ou de les modifier (dateAchat, marque, prixAchat et prixCourant). Il faut donc enrichir la classe Vehicule des "getters/setters" nécessaires :
public int getDateAchat(){ return dateAchat; } public String getMarque(){ return marque; } public double getPrixAchat(){ return prixAchat; } public void setPrixCourant(double prix){ prixCourant = prix; } |
Il aurait été possible de déclarer les attributs nécessaires comme protected dans la classe Vehicule pour s'éviter la peine de définir les getters/setters. Ceci peut cependant nuire à une bonne encapsulation : un autre programmeur peut hériter de votre classe Vehicule. Il aurait alors accès aux détails d'implémentation et vous ne pourriez plus modifier librement cette implémentation sans potentiellement causer du tort à ses programmes ! Voici enfin la méthode calculePrix pour les avions :
public void calculePrix(int anneActuelle) { double decote; if (moteur == "HELICES") { decote = 0.1 * heuresVol / 100.0; } else { decote = 0.1 * heuresVol / 1000.0; } setPrixCourant(Math.max(0.0, (1.0 - decote) * getPrixAchat())); } |
Pour finir, ceux qui le souhaitent peuvent trouver ici le code complet.
Le code complet est fourni ci-dessous.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 |
import java.util.Calendar; import java.util.ArrayList; /** * Classe principale */ class Direction { public static void main(String[] args) { Ecole epfl = new Ecole(); epfl.add(new EtudiantRegulier("Gaston Peutimide", 2020, "SSC", 6.0)); epfl.add(new EtudiantRegulier("Yvan Rattrapeur", 2016, "SSC", 2.5)); epfl.add(new EtudiantEchange("Bjorn Borgue", 2018, "Informatique", "KTH")); epfl.add(new Enseignant("Mathieu Matheu", 2015, "LMEP", 10000, "Physique")); epfl.add(new Secretaire("Sophie Scribona", 2005, "LMT", 5000)); epfl.afficherStatistiques(); epfl.afficherEPFLiens(); } } /** * La direction */ class Ecole { private ArrayList<EPFLien> gens; public Ecole() { gens = new ArrayList<EPFLien>(); } public void add(EPFLien personne) { if (personne != null) { gens.add(personne); } } /** * Cette méthode affiche l'ancienneté moyenne des personnes fréquentant l'école * et le nombre d'étudiants parmi eux */ public void afficherStatistiques() { System.out.println("Parmi les " + gens.size() + " EPFLiens, " + EPFLien.getNbEtudiants() + " sont des etudiants."); System.out.println("Ils sont ici depuis en moyenne " + ancienneteMoyenne() + " ans"); } // Cette méthode affiche les caractéristiques des personnes fréquentant l'école public void afficherEPFLiens() { System.out.println("Liste des EPFLiens: "); for (EPFLien epflien : gens) epflien.afficher(); } //un méthode utilitaire pour l'affichage des statistiques // (ne fait pas partie de l'API) private double ancienneteMoyenne() { if (gens.size() == 0) return 0.0; int anneeCourante = Calendar.getInstance().get(Calendar.YEAR); double moyenne = 0.0; for (EPFLien epflien : gens){ moyenne += (anneeCourante - epflien.getAnnee()); } return moyenne / gens.size(); } } /** * Les personnes fréquentant l'EPFL */ class EPFLien { private String nom; private int annee; // un compteur statique pour compter le nombre d'étudiants private static int nombreEtudiants; public EPFLien(String nom, int annee) { this.nom = nom; this.annee = annee; } //Cette méthode affiche les caractéristiques générales d'un EPFLien public void afficher() { System.out.println(" Nom : " + getNom()); System.out.println(" Annee : " + getAnnee()); } public String getNom() { return nom; } public int getAnnee() { return annee; } public static int getNbEtudiants() { return nombreEtudiants; } protected void ajoutEtudiant() { ++nombreEtudiants; } } /** * Les étudiants */ class Etudiant extends EPFLien { private String section; public Etudiant(String nom, int annee, String section) { super(nom, annee); this.section = section; ajoutEtudiant(); } public void afficher() { super.afficher(); System.out.println(" Section : " + getSection()); } public String getSection() { return section; } } /** * Les étudiants régulier */ class EtudiantRegulier extends Etudiant { private double moyenne; public EtudiantRegulier(String nom, int annee, String section, double moyenne) { super(nom, annee, section); this.moyenne = moyenne; } public void afficher() { System.out.println("Etudiant regulier:"); super.afficher(); System.out.println(" Moyenne : " + moyenne); } } /** * Les étudiants d'échange */ class EtudiantEchange extends Etudiant { private String uniOrigine; public EtudiantEchange(String nom, int annee, String section, String uniOrigine) { super(nom, annee, section); this.uniOrigine = uniOrigine; } public void afficher() { System.out.println("Etudiant d'echange:"); super.afficher(); System.out.println(" Uni d'origine : " + getUniOrigine()); } public String getUniOrigine() { return uniOrigine; } } /** * Le personnel de l'EPFL */ class Personnel extends EPFLien { private String labo; private int salaire; public Personnel(String nom, int annee, String labo, int salaire) { super(nom, annee); this.labo = labo; this.salaire = salaire; } public void afficher() { super.afficher(); System.out.println(" Laboratoire : " + getLabo()); System.out.println(" Salaire : " + getSalaire()); } public String getLabo() { return labo; } public int getSalaire() { return salaire; } } class Enseignant extends Personnel { private String section; public Enseignant(String nom, int annee, String labo, int salaire, String section) { super(nom, annee, labo, salaire); this.section = section; } public void afficher() { System.out.println("Enseignant:"); super.afficher(); System.out.println(" Section d'enseignement : " + getSection()); } public String getSection() { return section; } } class Secretaire extends Personnel { public Secretaire(String nom, int annee, String labo, int salaire) { super(nom, annee, labo, salaire); } public void afficher() { System.out.println("Secretaire:"); super.afficher(); } } |
Une version possible du corrigé est fournie ci-dessous.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 |
/* Classe pour représenter le courrier */ class Courrier { // retourne le montant n'ecessaire pour affranchir le courrier // en mode d'exp'edition normal // on va faire une chose tre`s vilaine parcequ'on ne connait pas les // m'ethodes abstraites : on va lui donner un corps arbitrairement // d'efini (car on ne sait pas la d'efinir proprement // a` ce niveau de la hi'erarchie public double affranchirNormal(){return 0;}; // la bonne solution consiste a` d'eclarer cette m'ethode comme suit: // abstract private double affranchirNormal(); // lorsque vous aurez vu les cours de la semaine prochaine, expliquez pourquoi... // les attributs (communs aux lettres et colis): private double poids; private boolean express; private String adresse; // un constructeur possible pour la classe public Courrier(double poids, boolean express, String adresse) { this.poids = poids; this.express = express; this.adresse = adresse; } // un getter pour le poids (car utile dans les sous-classe) public double getPoids() { return poids; } // retourne le montant n'ecessaire pour affranchir le courrier. // elle appelle affranchirNormal et retourne le double de ce montant // si le mode d'exp'edition est express ('eviter la duplication du code // qui double le montant dans les m'ethodes affranchir-normal // des sous-classes) public double affranchir() { if (! valide()) { return 0; } else { double total = affranchirNormal(); if (express) { total *= 2; } return total; } } // un courrier est invalide si l'adresse de destination est vide // methode utilis'ee par Boite::affranchir et // Boite::courriersInvalides public boolean valide() { return adresse.length() > 0; } @Override public String toString() { String s = ""; if (!valide()) { s+= "(Courrier invalide)\n"; } s+= " Poids : " + poids + " grammes\n"; s+= " Express : " + (express ? "oui" : "non") + "\n"; s+= " Destination : " + adresse + "\n"; s+= " Prix : " + affranchir() + " CHF\n"; return s; } } /* Une classe pour repr'esenter les lettres */ class Lettre extends Courrier { //attributs sp'ecifiques aux lettres: private String format = ""; public Lettre(double poids, boolean express, String adresse, String format){ super(poids, express, adresse); this.format = format; } // red'efinit affranchirNormal() public double affranchirNormal() { double montant = 0; if (format.equals("A4")){ montant = 2.0; } else { montant = 3.5; } montant += getPoids()/1000.0; return montant; } // inutile de red'efinir la méthode valide() pour les lettres @Override public String toString() { String s = "Lettre\n"; s += super.toString(); s += " Format : " + format + "\n"; return s; } } /* Une classe pour repr'esenter les publicit'es */ class Publicite extends Courrier { public Publicite(double poids, boolean express, String adresse){ super(poids, express, adresse); } // redéfinit affranchirNormal() public double affranchirNormal() { return getPoids()/1000.0 * 5.0; } // inutile de red'efinir la méthode valide() pour les publicités @Override public String toString() { String s = "Publicité\n"; s += super.toString(); return s; } } /* Une classe pour repr'esenter les colis */ class Colis extends Courrier { //attributs sp'ecifiques aux colis: private double volume; public Colis(double poids, boolean express, String adresse, double volume) { super(poids, express, adresse); this.volume = volume; } // redéfinit affranchirNormal(); public double affranchirNormal() { // affranchit les colis selon une formule pr'ecise return 0.25 * volume + getPoids()/1000.0; } // ici il faut red'efinir (sp'ecialiser) la re`gle de validit'e des colis // un colis est invalide s' il a une mauvaise adresse //ou depasse un certain volume public boolean valide(){ return (super.valide() && volume <= 50); } @Override public String toString() { String s = "Colis\n"; s += super.toString(); s += " Volume : " + volume + " litres\n"; return s; } } /* Une classe pour repr'esenter la boite aux lettre */ class Boite { private Courrier[] contenu; private int index; // constructeur public Boite(int max) { contenu = new Courrier[max]; index = 0; } // la méthode demand'ee public double affranchir() { double montant = 0.0; for(int i=0; i < index; ++i){ Courrier c = contenu[i]; montant += c.affranchir(); } return montant; } public int size() { return index; } public Courrier getCourrier(int index) { if (index < contenu.length) return contenu[index]; else return null; } // autre m'ethode demandée dans l'interface // d'utilisation de la classe public int courriersInvalides() { int count = 0; for (int i = 0; i < index; i++) { if (!contenu[i].valide()) count++; } return count; } // difficile de fonctionner sans public void ajouterCourrier(Courrier unCourrier) { if (index < contenu.length){ contenu[index] = unCourrier; index++; } else { System.out.println("Impossible d'ajouter un nouveau courrier. Boite pleine !"); } } public void afficher() { for (int i = 0; i < index; i++) { System.out.println(contenu[i]); } } } // PROGRAMME PRINCIPAL (non demandé) class Poste { public static void main(String args[]) { //Cr'eation d'une boite-aux-lettres Boite boite = new Boite(30); //Creation de divers courriers/colis.. Lettre lettre1 = new Lettre(200, true, "Chemin des Acacias 28, 1009 Pully", "A3"); Lettre lettre2 = new Lettre(800, false, "", "A4"); // invalide Publicite pub1 = new Publicite(1500, true, "Les Moilles 13A, 1913 Saillon"); Publicite pub2 = new Publicite(3000, false, ""); // invalide Colis colis1 = new Colis(5000, true, "Grand rue 18, 1950 Sion", 30); Colis colis2 = new Colis(3000, true, "Chemin des fleurs 48, 2800 Delemont", 70); //Colis invalide ! boite.ajouterCourrier(lettre1); boite.ajouterCourrier(lettre2); boite.ajouterCourrier(pub1); boite.ajouterCourrier(pub2); boite.ajouterCourrier(colis1); boite.ajouterCourrier(colis2); System.out.println("Le montant total d'affranchissement est de " + boite.affranchir()); boite.afficher(); System.out.println("La boite contient " + boite.courriersInvalides() + " courriers invalides"); } } |
Dernière mise à jour: 08/11/2024 (Revision: 1.2)