Java est à la fois un langage de programmation informatique orienté objet et un environnement d'exécution informatique portable créé par James Gosling et Patrick Naughton employés de Sun Microsystems avec le soutien de Bill Joy (cofondateur de Sun Microsystems en 1982), présenté officiellement le 23 mai 1995 au SunWorld1).
Le JSDK est le principal produit de Sun conscernant Java. Il est téléchargeable gratuitement2).
Le JSDK comprend
javacPATH du sstèmeCLASSPATH indique la liste des répertoires contenant des bibliothèques JavaJAVA_HOME indique le répertoire dans lequel Java est installé
Quand on a terminé d'éditer un programme Java (MyProgram.java), on le passe au compilateur qui le transforme en ByteCode (MyProgram.class). Une fois le fichier « compilé » en ByteCode, ce dernier doit être interprété par une marchine virtuelle, disponible sous différentes plateformes.
(graphique)
En java on peut écrire deux types de programmes
Point, Employé, Bouton, CompteEpargnesalaire, premierEmployeur, dateDeNaissance, calculPaie(), getSurface(), printDateDeNaissance // ====== CONSTRUCTEURS ========================================================== // // ====== GETTERS ================================================================ // // ====== SETTERS ================================================================ // // ====== MÉTHODES =============================================================== //
Il est conseillé de documenter son code avec javadoc.
//Fichier : Bienvenue.java /**Ceci est un commentaire javadoc.*/ public class Bienvenue { /* Comme en C/C++, la fonction main() est le point d’entrée de l'application. */ public static void main(String s[]) { System.out.println("Bonjour et Bienvenue"); } }
//Fichier : Montant.java public class Montant { public static void main(String s[]) { final double TPS = 0.07; final double TVQ = 0.075; double prixUnit=10; //le prix unitaire. int quantite=10; double mHt, mTps, mTvq, mTot; mHt = prixUnit * quantite; mTps = mHt * TPS; mTvq = (mHt + mTps)*TVQ; mTot = mHt + mTps + mTvq; System.out.println("Montant hors taxes : "+mHt+ "\nMontant TPS : "+mTps+"\nMontant TVQ : "+mTvq+ "\nMontant total : "+mTot); } }
Il y a deux grandes catégories de types de données en Java
Les types de données primitifs sont indiqués tout en minuscules et les types de données composites sont nommés avec la première lettre en majuscules (tout comme la convention pour nommer les classes).
| Type | Catégorie | Espace mémoire | Précision |
|---|---|---|---|
| byte | entier | 1 octet | -128 → 127 |
| short | entier | 2 octets | -32768 → 32767 |
| int | entier | 4 octets | 2 milliards |
| long | entier | 8 octets | -264 → 264-1 |
| float | réel | 4 octets | 3.40282347E+38 |
| double | réel | 8 octets | 1.79769313486231570E+308 |
| char | caractère | 2 octets | 65535 codes (unicode) |
| Classe | Catégorie |
|---|---|
| Integer | Entier |
| Long | Entier |
| Float | Réel |
| Double | Réel |
| Caracter | Caratère |
| Boolean | Booléen |
Quelques exemples pour déclarer des variables ou des objets.
Syntaxe générale :
type nomVariable;
Délaration - Initialisation :
type nomVariable = valeur;NomClasse nomVariable=new NomClasse(…);;Les variables (non locales à une méthode) sont automatiquement initialisées (0 pour les numériques et caractère, null pour les objets, false pour les booléens).
Liste des opérateurs par priorité décroissante :
| # | Opérateur |
|---|---|
| 1 | [] . () |
| 2 | ! ~ ++ – + - (cast) new |
| 3 | * / % |
| 4 | + - |
| 5 | << >> >>> |
| 6 | < <= > >= instanceof |
| 7 | == != |
| 8 | & |
| 9 | ^ |
| 10 | | |
| 11 | && |
| 12 | || |
| 13 | ?: |
| 14 | = += -= *= /= %= &= |= ^= <<= >>= >>>= |
Assocativité :
Aussi appelé ternary operator.
boolean expression ? valeurSiVrai : valeurSiFaux
Exemple :
String humeur = estDeBonneHumeur ? "Je suis content" : "Je ne suis pas content";
switch (variable) { case valeur1 : Liste d'instructions break; case valeur2 : Liste d'instructions break; case valeurn : Liste d'instructions break; default: Liste d'instructions break; }
Le code suivant fait une sorte de foreach en Java. Attention, dès que ce code trouve un élément avec isSelected à true, il le supprime et sort de la boucle for.
for (ColoredBox coloredBox: scene.coloredBoxes) if (coloredBox.isSelected) scene.coloredBoxes.removeElement(coloredBox);
int[] t1; // tableau d'entiers int t2[]; // " " int[] x, y; // x et y sont des tableaux int x, y[]; // seleuement y est un tableau Point[] t3; // tableau de Point (objets)
t1 = new int[20]; t3 = new Point[10];
La programmation orienté objet comprend plusieurs concepts :
private, public, protected)
Le nom de la classe et le nom de fichier sont liés, c'est-à-dire qu'une classe Etudiant doit avoir comme nom de fichier Etudiant.java.
Structure générale d'une classe
| <nom de la classe> | |
|---|---|
| <attributs> | Décrivent l'état (habituellement private) |
| <méthodes> Constructeurs | Instancier l'objet |
| <méthodes> | Décrivent le comportement (public) |
public class Etudiant { private String nom, prenom, adresse; private int age; // Constructeurs public Etudiant(String n, String name, String pname) { nas = n; nom=name; prenom=pname; age=16; adresse="Montréal"; } public void demenage (String nouvelleAdr) { adresse = nouvelleAdr; } }
Dans un programme Java :
Etudiant e1 = new Etudiant ("123123123", "Pierre", "Jean"); Etudiant e2; e2 = e1;
Les variables e1 et e2 sont des références (synonymes à pointeur) qui sont liés à un objet. Quand on fait e2 = e1;, l'adresse mémoire de l'objet e1 est copiée dans la variable e2.
public Class ClasseHote { [...] private class ClasseInterne { [...] } }
Méthodes d'accès :
getValeur ira chercher la valeur de l'attribut valeur.setValeur donnera une nouvelle valeur à l'attribut valeur. La surcharge (overload) d'une méthode c'est d'implémenter une nouvelle méthode ayant le même nom et étant dans la même classe, mais avec une signature différente, ou plutôt avec des paramètres différents.
La surdéfinition (overriding) c'est d'implémenter une méthode différemment qui a été héritée par une classe parent.
Le this fait référence à l'objet lui-même. On peut utiliser la référence this dans une méthode quand un identificateur fourni en paramètre a le même nom qu'un attribut :
public class Personne { private String nom; public void setNom (String nom) { this.nom = nom; } }
On peut aussi utiliser this pour surcharger le constructeur le l'objet :
public class Point { private double abscisse, ordonnee; public Point(double a, double b) { abscisse=a; ordonnee=b; } public Point(double a) { this(a,0); } public Point() { this(0); } }
Il est plus simple par la suite de modifier le code pour ajouter un attribut :
public class Point { private double abscisse, ordonnee; public Point(double a, double b) { this(a,b,0); } public Point(double a) { this(a,0); } public Point() { this(0); } public Point (double a, double b, double c) { abscisse = a; ordonnee = b; profondeur = c; } }
La classe Adresse :
public class Adresse { private String numRue, nomRue, codePostal, ville; public Adresse (String n, String r, String c, String v) { numRue = n; nomRue = r; codePostal = c; ville = v; } public void setVille (String v) { ville = v; } public String getVille () { return ville; }
La classe Personne :
public class Personne { private String nas, nom, prenom; public Adresse adresse; public Adresse getAdresse () { return adresse; } }
Personne a = new Personne(); // afficher à l'écran la ville ou habite la personne a Adresse adr = a.getAdresse(); String v= adr.getVille(); System.out.println(v); // ou bien : System.out.println(a.getAdresse().getVille());
Les données primitives sont toujours transmises par valeur.
public void incrementer (int x) { x++; } // ... int a=8; incrementer(a); System.out.println("a="+a); // affiche : a=8
Les objets sont toujours transmis par référence. Les références sont transmises par valeur.
public class Point { private double abscisse, ordonnee; public void setAbs(double a) { abscisse = a; } } public void changerAbs (Point x, double abs) { x.setAbs(abs); } public void permute (Point x, Point y) { Point tmp=x; x=y; y=tmp; }
Point p = new Point(); chagerabs(p,35); //l'abscisse de p vaut 35
Quand on
Point p = new Point(), q = new Point(); permute(p,q);
Les packages en Java sont des regroupement de classes.
Exemples de packages :
Avec javax.swing.JFrame, le fichier de classe (JFrame.class) se trouve dans un dossier javax que celui-ci contient un sous-dossier swing. Cette structure ne se trouve pas nécessairement sur le disque-dur, il peut se trouver dans un fichier JAR.
L'emplacement des packages n'est pas important, l'important c'est de vérifier la variable d'environnement CLASSPATH.
import javax.swing.*; public class Programme { public static void main(String s[]) { JFrame fen = new JFrame ("Titre"); } }
package com.sgariepy.geometrie; // toujours la premiere ligne du code // éventuels import public class Cercle { }
La convention dans le choix de noms packages est d'utiliser le nom de domaine inversé. Par exemple ibm.com nommera ses packages com.ibm.*.
La visibilité, du moins visible au plus visible :
| Nom | Notation | UML |
|---|---|---|
| Privé | private | - |
| Protégé | protected | # |
| Paquetage | package | ~ |
| Public | public | + |
L'accès par défaut (accès friendly ou accès package).
Avec la déclaration
Package 1 :
public class Classe1 { private int a; int b; Class2 x; // ok } class Class2 { Class1 u = new Class1(); u.b = 36; //ok }
Package 2 :
public class Classe3 { Class2 y; // non, car Class2 qui a un acces "protegé" n'est pas visible de l'exterieur de son package }
Ayant une classe mère Personne et une classe fille Étudiant. Dans Personne si l'attribut NAS est private la classe fille n'a pas accès. En ayant un attribut nom qui est protected, la classe Étudiant aura accès. Si aucun spécificateur d'accès n'est spécifié dans la classe mère pour un attribut prenom, alors cet attribut est accessible seulement si les deux classes sont du même package.
public class Groupe { private Etudiant[] etudiants; private int nbEtudiants; private static int capaciteMaximale = 20; // Note1 public static int getCapaciteMaximale () { // Note2 return capaciteMaximale; } }
Note1 → L'attribut capaciteMaximale restera pour tous les instances de la classe. Une modification se fait partout.
Note2 → Pour accéder à l'attribut static sans instance de classe, il faut que la méthode soit elle-même static. Les méthodes static ne peuvent pas toucher aux attributs non static.
static void main(...) { Groupe g1 = new Groupe(), g2 = new Groupe(), g3 = new Groupe(); }
public class Taxes { public static final double TPS = 0.05; // Note3 public static final double TVQ = 0.075; }
double montantHT = 300; double montantTPS; montantTPS = montantHT * Taxes.TPS;
Note3 → Comme les attributs de la classe Taxes sont finaux (final), il est possible, sans problème de les mettre public. Ensuite, dans un programme, on pourra accéder directement à l'attribut : Taxes.TPS.
Autre exemple : la classe Math → Math.PI.
String chaine1 = "abcd"; String chaine2 = "xyz"; if (chaine1.compareTo(chaine2)==0) { // logique }
| Résultat | Signification |
|---|---|
| <0 | La chaine se trouve avant, lexicographiquement |
| == 0 | Les deux chaînes sont équivalentes, lexicographiquement |
| >0 | La chaine se trouve après, lexicographiquement |
public class Personne { private String nas, nom; private int age; public int getAge { return age; } public String getNAS { return NAS; } }
static public void main () { Personne = new Personne (); int a p.getAge(); a -= 10; // l'age de la persone nest pas modifiee String s = p.getNAS(); // cette fonction retourne une référence vers la chaine du NAS de la personne, mais la string est inchangeable }
Pour travailler sur des chaînes de caractère altérables, il faut utiliser StringBuffer.
IndexOutOfBoundsException.int[] T; //ou : int T[]; Point[] TP; //ou : Point TP[];
T = new int[5]; TP = new Point[3];
Attention! Les objets contenus dans TP ne sont pas encore créés:
TP[1].setAbs(25); //NON T[1] = 26 //OK
Les tableaux étant des objets, l'affectation copie les références.
Pour la copie on utilise la méthode arrayCopyde la
classe System :
System.arrayCopy(T1,0,T2,0,3); System.arrayCopy(T1,debutsource,T2,debutdest,nbElements);
java.util.Arrays contient plusieurs méthodes pour la gestion de tableaux(copie, recherche binaire, test d'égalité,…). java.util.Arrays contient la méthode sort() qui trie un tableau en utilisant l'algorithme quicksort.double mat[][] = new double[4][3];//matrice 4x3.
double mat[][] = {{3,6,2},{7,1,5},{11,9,54},{1,5,6}};
mat est un tableau de taille 4 dont chaque élément est un tableau de 3 doubles. double tem[] = mat[i]; mat[i] = mat[j]; mat[j] = temp;
double M[][] = new double[4][]; for (i=0;i<M.length;i++) M[i] = new double[2*i+1];
Un vecteur (java.util.Vector) est un objet qui encapsule un tableau de taille variable. Ce tableau ne peut contenir que des objets.
<html><br/></html>
Avant JDK 1.5
Vector v = new Vector();
Dès la création du vecteur, une place pour le tableau est réservé en mémoire.
Point p = new Point(5,2); v.add(p); Integer x = new Integer(48); v.add(x); String nom = "Bernard"; v.add(nom);
La méthode addElement() fait le même travail que add().
| Contenu du vecteur v | Index |
|---|---|
| p de type Point | 0 |
| x de type Integer | 1 |
| nom de type String | 2 |
| … | 3 |
| … | 4 |
| … | 5 |
if (v.contains(q)); //q et étant deux Point ayant le meme abscisse/ordonnee, cela retourne vrai Object o = v.get(2); Object m = v.get(0); int a1 = p.getAbscisse(); //OK int a2 = m.getAbscisse(); // NON Point n = (Point) v.get(0); int a3 = m.getAbscisse(); //OK
<html><br/></html> À partir de JDK 1.5
Vector<Point> vp = new Vector<Point>(); Point p = ...; vp.add(p); Point r = vp.get(0); // inutile de faire un transtypage
<html><br/></html> La méthode toArray()
Cette méthode permet de retourner un tableau de références des objets.
Object[] = v.toArray(); ((Point) t[0]).setAbscisse(10); //premiere paranthèses pour forcer la priorité sur le cast
Exemple d'une personne. Une personne a plusieurs attributs (nas, nom, prenom, age, adresse) et méthodes (changerAdresse(String nouvelleAdr)). Un Étudiant est une (relation is a) Personne. Si un Étudiant n'a pas de NAS, il y a problème de conception, c'est-à-dire qu'elle n'est pas vraiment une personne.
Apports de l'héritage :
L'instance Étudiant qui dérive de Personne, n'a pas accès aux attributs de Personne : NAS, nom, prenom, etc. C'est pourquoi on utilise un spécificateur d'accès moins rigide que private : protected.
public class Etudiant extends Personne
public class Personne { protected String NAS, Nom, Prenom, Adresse; public void changerAdresse(string nouvAdr) { //... } public void print () { System.Out.print ("NAS:"+NAS+ "\nNom: "+nom+ "\nPrenom:"+prenom); } } public class Etudiant extends Personne { public void print () { super.print(); // apelle la méthode print de la classe mère System.Out.print ("NAS:"+NAS+ "\nNom: "+nom+ "\nPrenom:"+prenom); } public String getSession () { return session; } }
Une référence peut pointer sur différents types d'objets, selon s'ils sont hérités ou non. Conceptuellement, une Personne est une Personne, un Etudiant est un Etudiant et un Etudiant est une Personne. Par contre, une Personne n'est pas un Etudiant. Voir l'exemple suivant :
Personne p1, p2; Etudiant e1, e2; p1 = new Personne(); //OK e1 = new Etudiant(); //OK p2 = new Etudiant(); //OK e2 = new Personne(); //NON
Ce qui est intéressant, c'est si nous avons un tableau de références de type Personne, mais qui pointent vers différents types de personnes (Étudiants, Employés, Militaires, etc), la méthode print correspondant au type spécifique de l'objet (fils) est appelé.
String s1 = p1.getSession(); //NON String s2 = p2.getSession(); //NON - getSession n'est pas définie dans la classe mère (p2 est une référence Personne) - il faut un Cast String s2 = ((Etudiant) p2).getSession(); //OK - s'assurer que p2 fait bien référence à un étudiant (p2 instanceof Etudiant) String s3 = e1.getSession(); //OK
p1.print(); //OK, appel à print de personne p2.print(); //OK, appel de print de Etudiant (liaison dynamique) e1.print(); //OK, appel de print de Etudiant
On peut spécifier une méthode comme étant final. Cela ura pour effet qu'une classe fille dérivée de la classe mère ne pourra pas redéfinir cette méthode. Par exemple, la classe Personne contient une méthode getNAS(), on peut la mettre finale puisque sa seule et unique fonction est de retourner le NAS.
import java.util.*; public class Personne { public final String getNAS() { return nas; } } public class Etudiant extends Personne { public String getNAS() { //NON : getNAS() est finale. return nas; } }
Dans l'API, la classe classe String est finale.
Certaines méthodes sont héritées de classes parent et il peut être intéressant de les redéfinir.
Par exemple la méthode equals :
public boolean equals (Object p) { if (p instanceof Point) { Point t = (Point) p; return (x == t.getX() && y == t.getY()); } else return false; }
public class X { class Y { } }
Une classe interne (Y) a accès aux attributs privés de la classe « externe » (X). De cette façon, la classe X n'a pas accès aux attribut de la classe interne Y. On peut aussi spécifier des attributs public à la classe Y et spécifier cette même classe comme étant private. De cette façon, la classe X a accès aux données de la classe interne.
Exemple :
class Personne { ... public JPanel getUI() { return new PersonnePanel(); } class PersonnePanel extends JPanel { //création de l’interface graphique d’une //personne sous forme d’un panneau. } }
Puis, dans un programme :
JFrame fenetre x = new JFrame(); Personne x = new Personne(); JPanel xUI = x.getUI(); fenetre.getContentPane().add(xUI);
Exemple de classe abstraite :
public abstract class Figure { //Fichier:Figure.java public boolean getVisibilite() { return visible; } public void setVisibilite(boolean v) { visible = v; } public abstract double getSurface(); private boolean visible; }
Dans le cas suivant, on définit une classe qui hérite de la classe Figure (abstraite). On implémente la méthode getSurface().
public class Cercle extends Figure { //Fichier : Cercle.java public Cercle() { centre = new Point(0,0); rayon = 10; } public Cercle(double r) { centre = new Point(0,0); rayon = r; } public Cercle(Point c, double r) { centre = c; rayon = r; } public double getSurface() { return Math.PI*Math.pow(rayon,2); } private Point centre; private double rayon; }
La classe suivante est celle d'un Rectangle. On implémente aussi la méthode getSurface() de façon différente de Cercle.getSurface().
public class Rectangle extends Figure { //Fichier : Rectangle.java // ... public Rectangle(double L, double l) { centre=new Point(0,0); longueur=L;largeur=l; } public double getSurface() { return longueur*largeur; } public double getLongueur() { return longueur; } private Point sommetHG; private double longueur, largeur; }
Exemple de code avec ce qui est possible et impossible de faire :
class TestFigure { //Figure : TestFigure.java public static void main(String args[]) { Figure f1, f2, f3; f1 = new Figure(); //NON. Figure est abstraite f2 = new Cercle(); //OK System.out.println(f2.getSurface()); f3 = new Rectangle(9,3); System.out.println(f3.getSurface()); //OK System.out.println(f3.getLongueur());//NON System.out.println((Rectangle)f3.getLongueur());//NON System.out.println(((Rectangle)f3).getLongueur());//OK } }
L'interface est une classe abstraite encore plus généralisée. Elle sert
Création d'une interface :
interface MonInterface { public void f1(...); public double f2(...); }
Implémentation de l'interface :
public class ABC implements MonInterface { ... public void f1(...) { ... } public double f2(...) { ... } }
Propriétés d'une interface :
MonInterface x1, x2; x1 = new ABC();
if (x1 instanceof MonInterface) ...
public interface MonInterface2 extends MonInterface { public void f3(...); }
public class Util { public void trier (Cercle t[]) { for (int i = 0; i<t.lentgh-1; i++) for (int j=i+1; j<t.lentgh; j++) if (t[i].getSurface() > t[j].getSurface[]) { Cercle temp; temp = t[i]; t[i] = t[j]; t[j] = temp } } } // ... main () { Cercle t[] = new Cercle t[50]; // intanciation des 50 cercles // ... Util.trier(t); }
Pour réellement comparer des objets entre eux, plus qu'avec la méthode equals() qui retourne un boolean vrai ou faux, il faut utiliser la méthode compareTo() qui se trouve dans l'interface Comparable. La méthode compareTo() permet d'obtenir un ordre naturel avec les objets comparables. Par exemple, si l'on a plusieurs objets Personne qui ont un attribut âge, il serait intéressant de pouvoir les comparer à l'aide de cette caractéristique. Cette méthode retourne -1 si un objet a est plus petit que b, retourne 0 si un objet a est égal à b et retourne 1 si un objet a est plus grand que l'objet b.
Le comparable est utilisé pour établir un ordre naturel entre les instances d'un même type d'objet. Pour que le tout fonctionne, quelques étapes sont nécessaires :
Comparable → public class Figure implements Comparable {compareTo(Object o) là où c'est nécessaire (dans ce cas-ci, la classe Figure)Arrays.sort(t)On peut préciser la précision d'une comparaison en multipliant les valeurs comparables. Dans le code suivant, une précision de 1% est appliquée.
public int compareTo(Object o) { if (o instanceof Cercle) return (int) 100*getSurface() - 100*((Cercle)x).getSurface())); else
D'autres comparateurs que l'ordre naturel peuvent être définis. Par exemple un étudiant qui peut être ordonné par son nom, ou par sa note moyenne ou bien par son numéro d'étudiant.
public class Etudiant implements Comparable { private String num, nom, prenom; private double note; public int compareTo(Object x) { if (!(x instance of Etudiant)) throw new ClassCastException("message"); Etudiant e = (Etudiant) x; int res = nom.compareToIgnoreCase(e.nom); if (res != 0) return res; res = prenom.compareToIgnoreCase(e.prenom); return res; } public boolean equals (Object x) { if (!(x instance)) return false; return (this.compareTo(x)==0); } }
Fonctionne selon le modèle source-écouteur. Signifie qu'un composant graphique (bouton) qui est la source de l'événement. L'écouteur a une méthode de gestion de l'événement. Il s'enregistre comme écouteur auprès du bouton (composant/source). La source averti l'écouteur dès que l'événement se produit. Ensuite, l'écouteur exécute la méthode de la gestion de l'événement.
Le clic est un événement de type Action (rprésenté par la classe java.awt.event.ActionEvent.
L'écouteur d'un événement d'action est un objet, instance d'une classe, qui implémente l'interface ActionListener. Il faut d'abord créer une classe Ecouteur qui implémente l'interface ActionListener.
ActionListener contient une méthode public void actionPerformed(ActionEvent x).
Catégories d'événements :
Phases de mise en oeuvre d'événements
ActionListener (package java.awt.event)JButton b = new JButton(“OK”)b.addActionListener(e)e.actionPerformed()public class AppUI extends JFrame implements ActionListener { public AppUI () { // ajout des éléments graphiques miNouveau.addActionListener(this); miOuvrir.addActionListener(this); } // ... public void actionPerformed(ActionEvent e) { Object source = e.getSource(); if (source == miNouveau) { // instructions } else if (source == miOuvrir) { // instructions } } }
L'interface WindowListener contient plusieurs méthodes qui doivent être implémentés par les classes qui implémentent cette interface.
WindowClosed()WindowClosing()WindowActivated()WindowsDeactivated()WindowIconified()WindowDeconified()
Dans les applications, toutes ces méthodes ne sont pas nécessaires. On peut donc utiliser WindowAdapter qui implémente les méthodes vides de l'interface WindowListener. Donc, si notre classe descend de WindowAdapter, seulement les méthodes nécessaires seront implémentées.
Les applications créés en Java, devraient être implémentés avec le MVC (modèle, vue, contrôleur). Le modèle étant les classes d'objet (Figure, Cercle, Rectangle, Dessin, etc), les vues étant les interfaces graphiques qui affichent ou recuillent les informations sur les classes (modèle) et les contrôleurs qui incluent la gestion des événements. Les vues et les contrôleurs peuvent être regroupés en paires (ou plus) dans les mêmes classes (ex : ApplicationDessinUI).
Il y a donc trois types de classes selon ce modèle :
swing)
Les classes du modèle peuvent aussi implémenter l'interface Observable, c'est à dire qu'il y peut y avoir plusieurs vues qui seront alors mis à jour systématiquement lorsqu'une modification se fait à la classe observée. Les vues implémentent l'interface Observer et qui lui permettra justement d'observer une classe observable.
La connexion entre le modèle et la vue est rédigée à l'aide du patron de conception Observer. Pour mettre en place le patron, l'API Java fournit :
ObserverObservableQuelques méthodes peuvent être implémentées :
La principale méthode d'un observeur est utile quand les données du modèle ont changé et c'est → void update() qui fera la mise à jour de l'affichage selon les nouvelles données.