Java/Online-Bank
Auf dieser Seite sollen Sie anhand der Simulation eines Online-Kontos selbst Klassen definieren und die grundlegenden Elemente einer (objektorientierten) Programmiersprache
- Klasse
- Attribut
- Datentypen für Zahlen:
double
, für Zeichenketten:String
- Quelltext
- Instanz eines Objektes
- den Objektinspektor von BlueJ
und deren Umsetzung in Java mit Hilfe von BlueJ kennen lernen.
Einfache Theorie der Objektorientierten Programmierung
Zunächst klären wir am Beispiel Vogel, was eine Klasse ist: Ähnlich wie in der Biologie gruppiert man in der OOP (Objektorientierten Programmierung) Objekte und fasst sie in Klassen zusammen. Dabei stellt sich die Frage, was einen Vogel auszeichnet:
Ein Vogel hat bestimmte Eigenschaften (oder Attribute). Er hat zum Beispiel eine bestimmte Farbe, ein Geschlecht und man kann seine Flügelspannweite messen.
Ein Vogel kann bestimmte Dinge tun. Er kann etwa singen oder ein Ei legen. Das sind die sogenannten Methoden (Operationen) des Vogels (singe(...), legeEi(...)
).
Methoden teilt man in zwei Gruppen ein:
- Es gibt zum einen beobachtende oder auch sondierende Methoden. Diese beantworten z.B. die Frage: Welche Farbe hast du? Im Programm werden sie
gibFarbe()
odergetFarbe()
genannt. - Zum anderen gibt es Methoden, die Eigenschaften ändern können (verändernde Methode). Ein Beispiel wäre
setFarbe(“gruen“)
.
Aus der Klasse Vogel lassen sich konkrete Objekte bilden: z.B. vogel1, vogel2, vogel3. Jedes Objekt hat einen bestimmten Zustand. Sokann z.B. vogel1 die Farbe gelb haben, weiblich sein und 40cm Spannweite haben. Der Zustand von vogel2 könnte sein: gelb, 30 cm Spannweite, männlich. Während die Klasse Vogel etwas abstraktes ist (eine Art Bauplan für Objekte), ist ein Objekt konkret. Was das genau bedeutet, erfahren Sie, wenn Sie die folgenden Beispiele nachvollziehen.
Objekte sind in der objektorientierten Programmierung Daten (Eigenschaften oder auch Atrribute) und die damit verknüpfte Programmlogik (Methoden oder auch Operationen), die zu Einheiten, nämlich eben den Objekten, zusammengefasst sind.
Gleichartige Objekte werden zu Klassen zusammengefasst.
Klassen dienen als Vorlage (wie ein Bauplan) zur Herstellung von Objekten. Von einer Klasse können beliebig viele Objekte hergestellt werden. Die Objekte sind einzigartig, dasie einen unterschiedlichen Namen tragen müssen, obwohl ihr Zustand identisch sein kann.
Der Zustand (oder auch Status) ist die Gesamtheit der Werte der Eigenschaften.Klassendiagramm und Objektdiagramm
Im Klassendiagramm (auch UML-Klassendiagramm genannt) hält man die Elemente einer Klasse anschaulich fest: Klassenname, Eigenschaften und Methoden.
- Im Klassendiagramm werden Klassenname, Eigenschaften und Methoden festgehalten.
- Alle Methoden, die wir schreiben, sind öffentlich, also
public
, alle Eigenschaften sindprivate
Im Klassendiagramm bekommen Eigenschaften daher ein vorgestelltes–
und Methoden ein+
- Skat
- Was sind Eigenschaften der Klasse Kartenspiel?
- Was sind die Methoden?
- Legen Sie ein Klassendiagramm an.
- Nennen Sie Eigenschaften und Methoden des Objektes Kaffeemaschine.
- Legen Sie ein Klassendiagramm an.
Beispiel: Online-Bank
Wie sieht ein "Bauplan" zu einem Konto aus (z.B. Girokonto an einer Online-Bank)? Dazu beantworte ich die Frage, was ein Konto auszeichnet.
Die Klasse Konto
Eine Online-Bank benötigt zumindest den Namen des Besitzers und den aktuell vorhandenen Geldbetrag, um ein Konto anzulegen.
besitzerName
und kontostand
sind Attribute (auch Eigenschaften genannt) der Klasse Konto.
Ein Attribut ist ein strukturelles Merkmal einer Klasse.
- Es hat einen Namen und einen Typ.
- Konvention: Attributsnamen werden klein geschrieben.
Die Namen von zwei Attributen des Kontos haben wir bereits festgelegt: besitzerName und kontostand. Nun wird auch ein Typ (Datentyp) für jedes Attribut gefordert.
Java kennt verschiedene Datentypen. Zwei für unseren Zweck geeignete sind String
und double
.
Zeichenketten, also Aneinanderreihungen von Zeichen, werden in Java mit dem Datentypen String deklariert. Der Datentyp String eignet sich als Datentyp für besitzerName
, da dieser aus beliebigen Zeichen der Tastatur besteht und nicht etwa eine Zahl ist.
Als Typ des Kontostands kontostand bevorzugen wir einen numerischen Datentypen. double
kann Zahlen im Bereich von +/-1,7E+308 (also 17 mit 307 Nullen) abspeichern, was für unseren Zweck genügen sollte.
- Nennen Sie mögliche Attribute einer Klasse Auto, Uhr und Kaffeemaschine. Wählen Sie entsprechend Datentypen für die Attribute.
- Nennen Sie Fälle, bei denen der Zahlenbereich für
double
nicht ausreichend sein könnte.
Der erste Quelltext der Klasse Konto
Nach diesen Einführenden Überlegungen geht es nun daran, den Quelltext zu erarbeiten. Bei jedem Quelltext muss man sich an die Syntax der Programmiersprache richten. In Java beginnen wir damit, eine Klasse zu definieren. Dazu verwenden wir naheliegenderweise das Wort class.
Was bedeuten public und private?
public und private' regeln die Zugriffsrechte (Fachbegriff: Zugriffsmodifizierer). Vergleichbar ist das mit dem Zugriff auf eine Waschmaschine:
- Die Bedienknöpfe sind
public
, also öffentlich und von jedem bedienbar. - Das Innenleben ist
private
. Hier soll keine Hausfrau und erst Recht kein Hausmann Zugriff haben, da sie bzw. er etwas kaputtmachen könnte. Als erste Faustregel genügt es, sich zumerken, dass Attribute private (Innenleben der Klasse) und Klassen public sind (Jeder soll Klassen ausführen dürfen). Im Klassendiagramm zu Vogel haben wir bereits Zugriffsmodifizierer gesehen (siehe Konvention:Eigenschaften immer private, Klassen immer public)
Schritte zur Eingabe des Quelltexts
Nun kommt BlueJ ins Spiel. Mit "Projekt|neu" wird ein neues Projekt angelegt, das wir "Online-Bank" nennen wollen. Mit "Neue Klasse" legen wir eine Klasse mit dem Namen "Konto" an. Wenn alles richtig geklappt hat, sollte es etwa so aussehen:
Mit einem Doppelklick auf das beige Rechteck kommt zum Editor, in dem man den Quelltext eintippen kann und anschließend mit dem Button "Übersetzen" abspeichert und "compiliert".
public class Konto {
private String besitzerName;
private double kontostand;
}
Unten sollte nun zu sehen sein "Klasse übersetzt - keine Syntaxfehler". Wenn das nicht der Fall ist, beginnt die Suche nach Tippfehlern.
Nun sollten in dem Hauptfenster von BlueJ die Streifen von der Klassendarstellung verschwunden sein. Mit Hilfe der rechten Maustaste kann ich eine "Instanz des Objektes Konto" anlegen, indem ich new Konto()
wähle.
Ich wähle den vorgegebenen Instanznamen Konto1
und schon habe ich mein Objekt, welches BlueJ unten links als rotes Rechteck visualisiert.
Objektinspektor
Ein Rechtsklick auf das rote Rechteck und dann auf Inspizieren öffnet den Objektinspektor. Dieser gibt einen Überblick über den Zustand des Objektes: Unser Besitzer heißt null, ein Wort, das Java verwendet, wenn String-Variablen noch keinen Wert haben. Der Kontostand ist 0, was der Anfangswert für alle numerischen Variablen ist.
Zusammenfassung
- Eine Klasse (be-)schreibt man in BlueJ mit Hilfe eines Quelltextes. Der grundsätzliche minimale Aufbau ist:
public class Klassenname {
// Eigenschaften
private Datentyp eigenschaft1;
private Datentyp eigenschaft2;
}
- Die Zugriffsmodifizierer public und private regeln die Zugriffsrechte. Als vorläufige Regel ist die Klasse immer public und die Eigenschaften sind immer private.
- Aus Klassen lassen sich Objekte erzeugen, die einen eindeutigen Namen tragen müssen.
- Die Objekte haben einen Zustand, den man mit Hilfe des Objektinspektors ansehen kann. Anfangszustand von Strings ist das englische Wort
null
und von Zahlentypen0
. - Ein „Doppelslash“ // am Anfang der Zeile macht aus der Zeile einen Kommentar, den der Computer ignoriert. Ein Kommentar erläutert den Quelltext.
Nun möchten wir natürlich andere Werte setzen, einzahlen, auszahlen usw.. Das wird das Konto auch noch können, allerdings bedarf es dazu eines weiteren Ausbaus.
Zunächst daher noch eine
- Erweitern Sie die Klasse Konto um die Attribute
kreditlimit
undtelefonnummer
. Welche Datentypen würden Sie verwenden? - Entwerfen Sie das Klassendiagramm zur Klasse Konto. Entwerfen Sie den Quelltext zur Klasse Vogel.
- Profis: Recherchieren Sie zum Thema "Genauigkeit von Fließkommazahlen", nach den Grenzen der Genauigkeit des Typs double. Welche weiteren numerischen Datentypen bietet Java?
Konstruktor – Wie sind die Ausgangswerte?
Der Konstruktor ist eine spezielle Methode, die bei der Erzeugung von Variablen aufgerufen wird. Er wird dazu verwendet, die Variable in einen definierten Anfangszustand zu versetzen.Dieser Vorgang nennt sich auch Initialisierung.
Wir erweitern die Klasse Konto wie folgt:
public class Konto {
// (Instanzvariablen der) Eigenschaften
private String besitzerName;
private double kontostand;
//Konstruktor
public Konto (String pBesitzerName){
besitzerName = pBesitzerName;
kontostand = 1.0;
}
}
Begriff | Erläuterung |
---|---|
Konstruktor | legt die Anfangswerte fest. Er erhält den gleichen Namen wie die Klasse selbst: z.B. public Konto Ein Konstruktor ist eine spezielle Methode. Alle Werte, die nicht im Konstruktor festgelegt werden, bekommen einen „default-Wert“. Bei Zahlentypen ist das 0, bei Strings „null“ |
(Übergabe-)parameter | Sind die Werte, die der Methode übergeben werden. Für jeden Parameter muss ein Datentyp festgelegt werden. Der Name des Parameters kann zur besseren Übersicht mit p beginnen: z.B. pBesitzerName
|
null | Ist ein leeres Objekt. Es zeigt an, wenn eine Variable bzw. ein Objekt nicht mit einem Wert belegt wurde. |
geschweifte Klammern { } |
Tippt man mit Strg + Alt + 7 bzw. 0. Sie werden eingesetzt, um das Programm zu strukturieren.Die gesamte Klasse steht in geschweiften Klammern und auch Methoden wie der Konstruktor. Das Weglassen von Klammern ist eine typische Fehlerquelle.Das letzte Zeichen einer Klasse ist „}“ (vgl. oben) |
amerikanische Notation | Java ist amerikanisch, daher Punkt statt Komma: 3.141 |
Semikolon | Nach (nahezu) jedem Befehl (Anweisung) möchte Java ein Semikolon ; |
Mehrere Parameter
Möchte ich mehrere Parameter übergeben, so trenne ich sie mit einem Komma. Als Beispiel füge ich den Straßennamen hinzu.
public class Konto {
// Eigenschaften
private String besitzerName;
private String strassenName;
private double kontostand;
//Konstruktor
public Konto (String pBesitzerName, String pStrassenName){
besitzerName = pBesitzerName;
strassenName = pStrassenName;
kontostand = 1.0;
}
}
- Vollziehen Sie das angegebene Beispiel nach. Testen Sie Fehleingaben bei BlueJ: Lassen Sie beim Besitzernamen die Anführungstsriche weg, geben Sie eine Zahl, geben Sie nichts ein …
- Lassen Sie statt des Namens den Kontostand als Parameter übergeben.
- Ergänzen Sie die Eigenschaften
vorname
,telefonnummer
,postleitzahl
,geburtsjahr
,geschlecht
undhausnummer
. Begründen Sie dabei, welche Datentypen Sie verwenden.
Methoden – Was kann meine Klasse?
Die Methode einzahlen()
als Beispiel für eine verändernde Methode
Es sind natürlich noch viele weitere Methoden möglich. Ein weiteres Beispiel ist die Methode einzahlen()
.
public void einzahlen(doublepBtrag){
kontostand += pBetrag;
// oder kontostand = kontostand + pBetrag;
}<
Diese Methode sorgt dafür, dass zu dem bisherigen Kontostand ein bestimmter Betrag hinzugefügt wird. Hervorzuheben ist, dass die folgende Zeile nicht mathematisch zu verstehen ist:
kontostand = kontostand + pBetrag;
Das bedeutet: kontostand
wird die Summe aus dem bisherigen Kontostand und einem Parameter pBetrag
zugewiesen.
- Bauen Sie die Methode
einzahlen
in Ihre Kontoklasse ein. Testen Sie das Einzahlen von negativen Beträgen, großen Beträgen, 0, Wörtern, Rechnungen... - Entwerfen Sie auf Grundlage von
einzahlen
die Methodeauszahlen()
. - Entwerfen Sie
raubeAus()
, die den Kontostand auf 0 setzt.
Die set-Methoden als Bspl. für verändernde Methode ohne Rückgabewert
Diese einfache Methode verändert den Namen des Kontobesitzers. Eine Methode ohne Rückgabewert wird mit public void
(void entspricht im englischen etwa: nichts) eingeleitet.
public void setBesitzerName(String pNeuerName){
besitzerName = pNeuerName;
}
Die get-Methoden als Beispiel für sondierende Methoden
Eine ebenso einfache Methode liest den Namen des Kontobesitzers wieder aus:
public String getBesitzerName(){
return besitzerName;
}
Wann schreibe ich void?
void
kommt bei einer Methodendeklaration immer dann vor, wenn es keine Rückgabe gibt.return
Gibt es eine Rückgabe (return variable), so muss der Datentyp des Rückgabewertes eingetragen werden.
- Schreiben Sie die get- und set-Methode zu der Eigenschaft kontostand.
- Schreiben Sie die Methode
bucheZinsen (double zinssatz)
, die auf den vorliegenden Kontostandzinssatz
, Prozent, Zinsen addiert. - Zeichnen Sie das Klassendiagramm zu der aktuellen Klasse Konto.
Einige Beispiele
Wenn Sie bedenken, wie dick üblicherweise Java-Bücher sind, dann können Sie von den Beispielen zur Zeit noch nicht viel erwarten. Einige einfache Beispiele zeige ich Ihnen hier dennoch, damit Sie einen Überblick bekommen, wie eine Klasse in Java aufgebaut ist.
Der Zähler
Der Zähler ist eine einfache Klasse, bei dem Sie bei Aufruf der Methode erhoeheZaehlerstand()
den Zähler um 1 erhöhen.
Anwendungen: Verkehrszählungen, Heizungsverbrauchzähler, Bildzähler bei einer Kamera, Rundenzähler bei der Carrera-Bahn...
Bei den realen Anwendungen kommt der Impuls von einem Sensor bzw. einen Druckknopf. Bei der Simulation müssen Sie statt dessen eine Methode ausführen.
public class Zaehler
{
// Instanzvariablen der Eigenschaften
int zaehlerstand;
// Konstruktor
public Zaehler()
{
// Der Anfangszählerstand soll immer 0 sein.
zaehlerstand=0;
}
// Methoden
public void erhoeheZaehlerstand()
{
zaehlerstand += 1;
// oder zaehlerstand++;
// oder zaehlerstand=zaehlerstand+1;
}
}
Das Telefonbuch
Der Java-Editor half bei der Erstellung dieser Klasse. Unter UML|neue Klasse erhalten Sie ein praktisches Auswahl-Menü, das Ihnen viel Tipparbeit ersparen kann. Der Java-Editor zeigt übrigens im Klassendiagramm auch den Konstruktor an. Das wird in der Literatur nicht einheitlich gemacht, oftmals wird er nicht angegeben.Die Telefonnummer wird, obwohl sie laut Name eine Nummer ist, als String eingelesen, um Eingaben wie „/“ zwischen Vorwahl und Nummer zuzulassen. Die Besonderheit an dieser Klasse ist, dass name
nur durch den Konstruktor verändert werden kann. D.h. ich kann zwar mit setTelefonnummer
nachträglich Telefonnummern ändern, aber der Name bleibt erhalten. Allenfalls durch das Löschen des Objektes kann ich Personen aus dem Telefonbuch entfernen.
public class Telefonbuch {
// Eigenschaften
// Anfang Variablen
public String name;
public String telefonnummer;
// Ende Variablen
// Konstruktor
public Telefonbuch(String name, String telefonnummer) {
// ... Hier fehlen 2 Zeilen. Bitte nachtragen! (vgl. Übung)
}
// Anfang Ereignisprozeduren
// Methoden
public String getTelefonnummer() {
return telefonnummer;
}
public void setTelefonnummer(String pTelefonnummer) {
telefonnummer = pTelefonnummer;
}
public String getName() {
return name;
}
// Ende Ereignisprozeduren
}
Das Auto
Die folgende Klasse ist schon sinnvoller. Sie ist in ähnlicher Weise in vielen Bordcomputern von modernen Autos eingebaut: Die elementare Frage beim Autofahren ist: wie weit komme ich noch, ohne zu tanken. Die Klasse simuliert ein Auto, erfasst aber nur Daten um den Tank und den Verbrauch.
Besonderheit: tanke(..)
und fahre(...)
sind Methoden, die den aktuellen Füllstand beeinflussen. Es sind auch in der Realität die einzigen Arten, wie man den Füllstand eines Autos ändert (von klaueBenzin()
, tropfeHeraus()
und verdampfe()
abgesehen). Spannend ist die sondierende Methode getAktuelleMaximaleReichweite
, die den Wert in km zurückgibt, der bei aktuellem Füllstand noch gefahren werden kann. Die Probleme der Klasse liegen noch darin, dass man (auch über das Tankvolumen hinaus) soviel tanken kann wie man will und dass der Tankinhalt auch in den negativen Bereich fahren kann (Dispokredit bei der Tankstelle). Das können wir erst lösen, wenn wir die bedingte Ausführung kennen gelernt haben. Der folgende Quelltext ist aus dem Java-Editor übernommen.
public class Auto {
private double volumenTank;
private double verbrauchAuf100km;
private double aktuellerFuellstand;
// Konstruktor
public Auto(double pVolumenTank, double pVerbrauchAuf100km) {
volumenTank=pVolumenTank;
verbrauchAuf100km=pVerbrauchAuf100km;
aktuellerFuellstand=0; // Bei Auslieferung ist der Tank leer
}
public double getAktuellerFuellstand() {
return aktuellerFuellstand;
}
// Mathoden
public void setAktuellerFuellstand(double pAktuellerFuellstand) {
aktuellerFuellstand = pAktuellerFuellstand;
}
public double getVerbrauchAuf100km() {
return verbrauchAuf100km;
}
public double getVolumenTank() {
return volumenTank;
}
public void tanke(double menge) {
this.aktuellerFuellstand+=menge; // füge Menge hinzu
}
public void fahre(double pKilometerZahl) {
// Die Formel sagt folgendes aus:
// vom aktuellen Füllstand wird abgezogen: der Verbruch bei der Kilometerzahl
aktuellerFuellstand -= pKilometerZahl / 100 * verbrauchAuf100km;
}
}
- Analysieren und testen Sie alle 3 Programme. Warum bekommt nicht jede Eigenschaft eine get- und eine set-Methode?
- Zu Zähler:
- Ergänzen Sie die Methode: getZaehlerstand()
- Ergänzen Sie die Methode: aufNullSetzen() und erhöheUm(double pZahl)
- Zu Telefonbuch: Ergänzen Sie die fehlenden Zeilen zum Konstruktor.
- Ergänzen Sie die Eigenschaften geburtsdatum (ggf. noch schuhgröße). Welchen Datentypen verwenden Sie?
- Benötigt die Eigenschaft eine set-Methode?
- Zum Auto: Ergänzen Sie die Methode: getMaximaleReichweiteBeiVollemTank()
- Entwickeln Sie die Klasse Handy, die die Eigenschaften telefonnummer, guthaben undgespraechskostenProMinute. Entwickeln Sie geeignete Methoden.
- Übung für Fortgeschrittene: Realisieren Sie die Klasse Bruch mit den Eigenschaften zaehler und nenner und realisieren Sie die Methoden addierenEinerGanzenZahl, kuerzen und erweitern. Wenn Sie schon Vorwissen haben, können Sie hier für eine erweiterte Variante schauen: 2.04 Anwendungsbeispiel Bruchrechnung (drdanielappel.de)
Interaktion zwischen Objekten
Zum Schluss des Kapitels behandeln wir, wie zwei Objekte miteinander kommunizieren können. Ein schönes Beispiel dafür ist die Methode ueberweise()
:
public void ueberweise(double pBetrag, Konto pZielkonto){
abheben(pBetrag);
zielkonto.einzahlen(pBetrag);
}
Es gibt diesmal zwei Parameter, die wir, wie schon beim Konstruktor, mit Komma trennen. Die Besonderheit ist, dass der Datentyp von Zielkonto wieder ein Konto ist. Man darf also Klassen als Datentypen angeben – ja, man darf sogar in Methoden den Datentypen der Klasse verwenden, zu dem die Methode gehört. Das ist sinnvoll, denn üblicherweise möchte man von einem Konto auf ein anderes Konto überweisen. Also muss das Konto überweisen können (wenn man keine Klasse Bank erschafft, die die Konten verwaltet. Aber das ist ein Thema für später)
<source lang="java"> abheben(betrag); zielkonto.einzahlen(pBetrag); </source|
Eigenschaften, die private sind, sind nur für Objekte der gleichen Klasse sichtbar. D.h. theoretisch können wir von dem einen Konto auf die Eigenschaften des anderen Kontos zugreifen. Damit es nicht zu Verwechselungen kommt, sollten Sie sich das aber erst gar nicht angewöhnen.
Konvention:
Auf die Daten von fremden Objekten oder Klassen greifen wir immer über Methoden und nie über deren Eigenschaften zu. Dies gewährleistet, dass nichts Ungewolltes verändert wird (vgl. Waschmaschinenbeispiel). Nur innerhalb des Objektes selbst verändert man die eigenen Eigenschaften direkt.- Schreiben Sie die Methode
holeEinzug (double betrag, Konto zielkonto)
, bei dem das Geld von einem fremden Konto überwiesen wird. - Schreiben Sie die Methode
vererben (Konto erbe)
, bei dem das gesamte Geld eines Kontos auf ein anderes verbucht wird. - Schreiben Sie die Methode
binIchReich(Konto vergleichskonto)
, das die Differenz aus dem eigenen Kontostand mit einem anderen Konto vergleicht. - Schreiben Sie die Methode
stifteVerwirrung(Konto konto1, Konto konto2)
, bei der ein unbeteiligtes Konto dafür sorgt, dass das gesamte Geld vonkonto1
auf konto2 überwiesen wird. - Schreiben Sie die Methode
stifteNochMehrVerwirrung(Konto konto1, Konto konto2)
, bei der ein unbeteiligtes Konto dafür sorgt, dass das gesamte Geld vonkonto1
halbiert, das vonkonto2
aber verdoppelt wird. - Schreiben Sie die Methode
bekommeZinsen(double zinsatz)
, das entsprechend des Zinssatzes den Kontostand erhöht. - Lassen Sie von einem Konto den Namen des Besitzers eines anderen Kontos ändern. Welche Probleme ergeben sich, wenn Sie die Methode
setBesitzerName
weglassen?