Java Tutorial - Wilkening
Transcription
Java Tutorial - Wilkening
Objektorientiertes Programmieren in Java - V. 29 Seite 1 / 409 Objektorientiertes Programmieren in Java Detlef Wilkening www.wilkening-online.de © 1997-2016 Version 29 Dieses Java Tutorial darf in unveränderter Form für Unterrichtszwecke weitergegeben und verwendet werden. Der Autor freut sich aber über Rückmeldungen zum Einsatz des Tutorials. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 2 / 409 Objektorientiertes Programmieren in Java 1 Organisatorisches ......................................................................................................... 13 2 Java ................................................................................................................................ 14 2.1 Einleitung .................................................................................................................... 15 2.2 Java als Plattform........................................................................................................ 15 2.3 Java Versionen ........................................................................................................... 21 2.4 Technologien .............................................................................................................. 27 2.5 Fazit ............................................................................................................................ 30 3 Mini-Einführung ............................................................................................................. 30 3.1 Applikation .................................................................................................................. 31 3.2 Quelltext ...................................................................................................................... 32 3.3 Ausgabe ...................................................................................................................... 35 3.4 Strings ......................................................................................................................... 36 3.5 Arrays und Kommandozeilen-Argumente .................................................................... 36 3.6 Klassen ....................................................................................................................... 37 3.7 Funktionen .................................................................................................................. 37 3.8 Packages .................................................................................................................... 38 3.9 Exceptions .................................................................................................................. 39 3.10 Eingabe .................................................................................................................. 40 3.11 Konvertierungen ..................................................................................................... 43 4 Praktikum ....................................................................................................................... 43 4.1 Tools ........................................................................................................................... 43 4.2 Source Organisation ................................................................................................... 54 4.3 Benutzung der JDK Entwicklungs-Tools...................................................................... 54 4.4 Projekt Tools ............................................................................................................... 67 4.5 Source-Path und Class-Path ....................................................................................... 68 4.6 Eclipse ........................................................................................................................ 70 4.7 Projekte importieren .................................................................................................... 85 4.8 Aufgaben..................................................................................................................... 90 4.9 Lsg. zu Aufgabe „Hallo Welt“ – Kap. 4.8.1 .................................................................. 92 4.10 Lsg. zu Aufgabe „GUI-Fenster“ – Kap. 4.8.2 .......................................................... 93 4.11 Lsg. zu Aufgabe „Package“ – Kap. 4.8.3 ................................................................ 93 4.12 Lsg. zu Aufgabe „Packages“ – Kap. 4.8.4 .............................................................. 93 4.13 Lsg. zu Aufgabe „Summe“ – Kap. 4.8.5 .................................................................. 94 4.14 Lsg. zu Aufgabe „Ausgabe Verzeichnis“ – Kap. 4.8.6 ............................................. 97 5 Elementare Datentypen und Variablen ......................................................................... 98 5.1 Datentypen .................................................................................................................. 98 5.2 Elementare Datentypen............................................................................................... 99 5.3 Variablen ................................................................................................................... 101 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 3 / 409 5.4 Wert- und Referenz-Semantik ................................................................................... 102 6 Operatoren ................................................................................................................... 105 6.1 Zuweisungs-Operatoren ............................................................................................ 106 6.2 Gleich- und Ungleich-Operatoren .............................................................................. 106 6.3 Geteilt- und Modulo-Operator .................................................................................... 108 6.4 Bitweises AND, OR, XOR ......................................................................................... 110 6.5 Boolsches bzw. bedingtes AND, OR, XOR ............................................................... 110 6.6 Prä- und Post-Inkrement und -Dekrement Operatoren .............................................. 111 6.7 Schiebe-Operatoren .................................................................................................. 111 6.8 Frage-Zeichen Operator ............................................................................................ 112 6.9 Aufgaben................................................................................................................... 112 6.10 Lsg. zu Aufgabe „Inkrement-Operator“ – Kap. 6.9.1 ............................................. 112 7 Kontroll-Strukturen...................................................................................................... 113 7.1 Bedingter-Kontrollfluss – If ........................................................................................ 113 7.2 Mehrfach-Verzweigung – Switch ............................................................................... 117 7.3 For-Schleife ............................................................................................................... 119 7.4 For-Schleife für Container und Arrays ....................................................................... 120 7.5 While-Schleife ........................................................................................................... 120 7.6 Do-Schleife ............................................................................................................... 121 7.7 Break- und Continue-Anweisung ............................................................................... 121 7.8 Gelabelte Sprung-Anweisungen ................................................................................ 122 7.9 Aufgaben................................................................................................................... 123 7.10 Lsg. zu Aufgabe „Schleifen-Varianten“ – Kap. 7.9.1 ............................................. 125 7.11 Lsg. zu Aufgabe „Teilbar?“ – Kap. 7.9.2 ............................................................... 126 7.12 Lsg. zu Aufgabe „Lesbare Zahlen“ – Kap. 7.9.3 ................................................... 127 7.13 Lsg. zu Aufgabe „Zahlen-Liste“ – Kap. 7.9.4......................................................... 128 7.14 Lsg. zu Aufgabe „Mittelwert und Varianz“ – Kap. 7.9.5 ......................................... 131 7.15 Lsg. zu Aufgabe „Multiplikations-Matrix“ – Kap. 7.9.6 ........................................... 132 8 Funktionen ................................................................................................................... 134 8.1 Funktions-Arten ......................................................................................................... 135 8.2 Syntaktischer Aufbau ................................................................................................ 135 8.3 Variablen in Funktionen............................................................................................. 135 8.4 Funktions-Parameter ................................................................................................. 137 8.5 Funktions-Rückgaben ............................................................................................... 138 8.6 Überladen ................................................................................................................. 139 8.7 Rekursion .................................................................................................................. 140 8.8 Aufgaben................................................................................................................... 141 8.9 Lsg. zu Aufgabe „Fakultäts Funktion“ – Kap. 8.8.1.................................................... 143 8.10 Lsg. zu Aufgabe „Primzahl?“ – Kap. 8.8.2 ............................................................ 145 8.11 Lsg. zu „Aufgabe „Schleife rekursiv“ – Kap. 8.8.3 ................................................. 146 8.12 Lsg. zu Aufgabe „Zahlen-Liste 2“ – Kap. 8.8.4...................................................... 146 8.13 Lsg. zu Aufgabe „Quadratwurzel“ – Kap. 8.8.5 ..................................................... 147 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 4 / 409 9 Ausgewählte Bibliotheks-Klassen .............................................................................. 152 9.1 Klasse String ............................................................................................................. 153 9.2 Klassen StringBuilder & StringBuffer ......................................................................... 156 9.3 Container & Iteratoren ............................................................................................... 157 9.4 Wrapper Klassen ...................................................................................................... 176 9.5 Zufalls-Zahlen ........................................................................................................... 178 9.6 Datum und Uhrzeit .................................................................................................... 179 9.7 Datei- und Verzeichnis-Handling ............................................................................... 180 9.8 Aufgaben................................................................................................................... 187 9.9 Lsg. zu Aufgabe „String-Analyse“ – Kap. 9.8.1.......................................................... 192 9.10 Lsg. zu Aufgabe „Hallo <Person>“ – Kap. 9.8.2 .................................................... 196 9.11 Lsg. zu Aufgabe „Hallo <Personen>“ – Kap. 9.8.3 ................................................ 196 9.12 Lsg. zu Aufgabe „Lesbare Zahlen 2“ – Kap. 9.8.4 ................................................ 198 9.13 Lsg. zu Aufgabe „Lottozahlen“ – Kap. 9.8.5 .......................................................... 200 9.14 Lsg. zu Aufgabe „Telefonbuch“ – Kap. 9.8.6 ........................................................ 201 9.15 Lsg. zu Aufgabe „Platten-Platz Verbrauch“ – Kap. 9.8.7 ....................................... 202 9.16 Lsg. zu Aufgabe „Datei-Suche“ – Kap. 9.8.8......................................................... 203 10 Arrays ........................................................................................................................... 204 10.1 10.2 10.3 10.4 10.5 10.6 10.7 10.8 10.9 Deklaration ........................................................................................................... 204 Arrays haben ein Attribut: length........................................................................... 205 Arrays durchlaufen ............................................................................................... 206 Arrays sind Referenzen ........................................................................................ 206 Arrays können mehrdimensional sein ................................................................... 207 Fehlerhafter Index ................................................................................................ 208 Vergleiche ............................................................................................................ 208 Aufgaben .............................................................................................................. 209 Lsg. zu Aufgabe „Lesbare Zahlen 3“ – Kap. 10.8.1............................................... 209 11 Klassen ......................................................................................................................... 210 11.1 11.2 11.3 11.4 11.5 11.6 11.7 11.8 Motivation ............................................................................................................. 210 Klassen-Entwurf ................................................................................................... 211 Beispiel ................................................................................................................. 212 Klassen-Implementierung ..................................................................................... 215 Aufgaben .............................................................................................................. 220 Lsg. zu Aufgabe „Ringzähler“ – Kap. 11.5.1 ......................................................... 223 Lsg. zu Aufgabe „Kontaktdaten 1“ – Kap. 11.5.2 .................................................. 225 Lsg. zu Aufgabe „Kontaktdaten 2“ – Kap. 11.5.3 .................................................. 228 12 Klassen-Details ............................................................................................................ 229 12.1 12.2 12.3 12.4 12.5 Zugriffsbereiche .................................................................................................... 229 Konstruktoren ....................................................................................................... 230 finalize .................................................................................................................. 231 Funktionen ............................................................................................................ 232 Attribute ................................................................................................................ 235 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 12.6 12.7 12.8 12.9 12.10 12.11 Seite 5 / 409 Aufzählungen........................................................................................................ 236 Top-Level-Klassen ................................................................................................ 239 Aufgaben .............................................................................................................. 239 Lsg. zu Aufgabe „Türme von Hanoi“ – Kap. 12.8.1 ............................................... 244 Lsg. zu Aufgabe „Gerüchteküche“ – Kap. 12.8.2 .................................................. 246 Lsg. zu Aufgabe „Tic-Tac-Toe 1“ – Kap. 12.8.3 .................................................... 250 13 Packages ...................................................................................................................... 271 13.1 13.2 13.3 13.4 13.5 13.6 Package-Anweisung ............................................................................................. 271 Klassen in Packages ............................................................................................ 272 Language-Package .............................................................................................. 273 Verschachtelung ................................................................................................... 273 Default-Package ................................................................................................... 273 Verzeichnisse ....................................................................................................... 274 14 Vererbung..................................................................................................................... 274 14.1 14.2 14.3 14.4 14.5 14.6 14.7 14.8 14.9 14.10 14.11 14.12 14.13 14.14 14.15 14.16 14.17 14.18 14.19 14.20 14.21 14.22 14.23 14.24 Vererbungs-Hierarchien........................................................................................ 274 Implementation ..................................................................................................... 275 Schlüsselwort super.............................................................................................. 276 Konstruktoren ....................................................................................................... 276 finalize .................................................................................................................. 277 Überschreiben ...................................................................................................... 277 Ist-ein Beziehung .................................................................................................. 279 Polymorphie.......................................................................................................... 279 abstract................................................................................................................. 280 Casts und instanceof ............................................................................................ 281 Klasse „java.lang.Object“ ...................................................................................... 283 Anwendung – Beispiel „Obstkorb“ ........................................................................ 285 Interfaces.............................................................................................................. 293 Modul-Entkopplung ............................................................................................... 294 Fazit...................................................................................................................... 297 Aufgaben .............................................................................................................. 297 Lsg. zu Aufgabe „Obstkorb“ – Kap. 14.16.1 .......................................................... 299 Lsg. zu Aufgabe „ProgressBar“ – Kap. 14.16.2 .................................................... 299 Lsg. zu Aufgabe „Kontaktdaten 3“ – Kap. 14.16.3 ................................................ 300 Lsg. zu Aufgabe „Kontaktdaten 4“ – Kap. 14.16.4 ................................................ 303 Lsg. zu Aufgabe „Tic-Tac-Toe 2“ – Kap. 14.16.5 .................................................. 303 Lsg. zu Aufgabe „Tic-Tac-Toe 3“ – Kap. 14.16.6 .................................................. 306 Lsg. zu Aufgabe „Tic-Tac-Toe 4“ – Kap. 14.16.7 .................................................. 307 Lsg. zu Aufgabe „Tic-Tac-Toe 5“ – Kap. 14.16.8 .................................................. 308 15 Innere Klassen ............................................................................................................. 308 15.1 15.2 15.3 Eingebettete static Klassen .................................................................................. 309 Eingebettete nicht static Klassen .......................................................................... 310 Eingebettete Interface’s ........................................................................................ 311 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 15.4 15.5 15.6 Seite 6 / 409 Lokale Klassen ..................................................................................................... 311 Anonyme Klassen ................................................................................................. 313 Virtuelle Maschine ................................................................................................ 314 16 GUI Programmierung mit Swing................................................................................. 314 16.1 16.2 16.3 16.4 16.5 16.6 16.7 16.8 16.9 Ein einfaches Fenster ........................................................................................... 316 Ein etwas besseres Fenster.................................................................................. 317 Ein grafisches „Hallo Welt“ ................................................................................... 318 Graphics Objekt .................................................................................................... 320 Klasse „Color“ ....................................................................................................... 321 Änderung der Oberfläche in den Windows-Style .................................................. 322 Aufgaben .............................................................................................................. 324 Lsg. zu Aufgabe „Schwebende Kugel“ – Kap. 16.7.1 ............................................ 325 Lsg. zu Aufgabe „Farb-Fenster“ – Kap. 16.7.2...................................................... 325 17 Philosophie der GUI Programmierung ....................................................................... 325 17.1 17.2 Besitzer des Bildschirms ....................................................................................... 325 Kontrollfluß ........................................................................................................... 326 18 Event-Modelle von Swing............................................................................................ 327 18.1 18.2 18.3 18.4 18.5 18.6 Events und Gruppierungen ................................................................................... 327 Internes Event-Modell ........................................................................................... 328 Externes Event-Modell .......................................................................................... 333 Vergleich der Event-Modelle ................................................................................. 345 Scribble 4 mit Daten-Modell .................................................................................. 345 Aufgaben .............................................................................................................. 348 19 Swing Layouts ............................................................................................................. 348 19.1 19.2 19.3 19.4 19.5 19.6 19.7 Swing Klasse „JButton“......................................................................................... 349 Grundlagen ........................................................................................................... 349 Layouts ................................................................................................................. 350 Verschachtelte Layouts ........................................................................................ 353 Aufgaben .............................................................................................................. 354 Lsg. zu Aufgabe „Layouts 1“ – Kap. 19.5.1 ........................................................... 354 Lsg. zu Aufgabe „Layouts 2“ – Kap. 19.5.2 ........................................................... 354 20 Swing GUI-Elemente.................................................................................................... 354 20.1 20.2 20.3 20.4 20.5 20.6 20.7 20.8 20.9 Labels ................................................................................................................... 354 Text-Felder ........................................................................................................... 355 Buttons ................................................................................................................. 357 Radio-Buttons ....................................................................................................... 358 Check-Boxen ........................................................................................................ 359 Scroll-Bars und Scroll-Panes ................................................................................ 361 Tabellen................................................................................................................ 361 Panels .................................................................................................................. 365 Menüs................................................................................................................... 365 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 20.10 20.11 20.12 20.13 20.14 20.15 Seite 7 / 409 Timer .................................................................................................................... 367 Aufgaben .............................................................................................................. 369 Lsg. zu Aufgabe „Liste“ – Kap. 20.11.1................................................................. 370 Lsg. zu Aufgabe „Scribble 6“ – Kap. 20.11.2 ........................................................ 370 Lsg. zu Aufgabe „TextField“ – Kap. 20.11.3.......................................................... 370 Lsg. zu Aufgabe „Kontaktdaten 5“ – Kap. 20.11.4 ................................................ 370 21 Applets ......................................................................................................................... 370 21.1 21.2 21.3 21.4 21.5 21.6 21.7 21.8 Beispiel ................................................................................................................. 370 Applet HTML-Seite ............................................................................................... 371 Grundlagen ........................................................................................................... 373 Applet-Parameter ................................................................................................. 374 Applets & Applikationen ........................................................................................ 375 Aufgaben .............................................................................................................. 376 Lsg. zu Aufgabe „Statuszeilen-Applet“ – Kap. 21.6.1 ............................................ 376 Lsg. zu Aufgabe „Scribble 7“ – Kap. 21.6.2 .......................................................... 376 22 Exceptions ................................................................................................................... 376 22.1 22.2 22.3 22.4 22.5 22.6 22.7 22.8 22.9 22.10 22.11 22.12 Motivation ............................................................................................................. 376 Realisation ............................................................................................................ 377 Fehler über mehrere Funktions-Ebenen ............................................................... 378 Aufräumarbeiten und „finally“ ................................................................................ 379 Exception-Objekte und Exception-Hierarchie........................................................ 381 Fangen von Basis-Exception-Klassen .................................................................. 382 Mehrere catch-Blöcke ........................................................................................... 383 Unbehandelte Exceptions ..................................................................................... 384 Geprüfte und ungeprüfte Exceptions .................................................................... 385 Verschachtelte try-Blöcke ..................................................................................... 387 Exception-Objekte ................................................................................................ 390 Exception-Sicherheit ............................................................................................. 391 23 Streams ........................................................................................................................ 393 23.1 23.2 23.3 23.4 Einführung ............................................................................................................ 393 Java Streams........................................................................................................ 393 Beispiele ............................................................................................................... 397 Stream-Tokenizer ................................................................................................. 399 24 Reflexion ...................................................................................................................... 400 24.1 24.2 24.3 24.4 24.5 Objekt-Erzeugung über den Klassen-Namen ....................................................... 400 Klasse „Class“ ...................................................................................................... 402 Disassemblieren ................................................................................................... 404 Aufgaben .............................................................................................................. 404 Lsg. zu Aufgabe „Klassen-Inspektor“ – Kap. 24.4.1 .............................................. 404 25 Serialisierung ............................................................................................................... 405 25.1 Interface Serializable ............................................................................................ 405 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 25.2 25.3 25.4 25.5 25.6 25.7 Seite 8 / 409 Rekursion ............................................................................................................. 406 Klassen-Variablen................................................................................................. 408 Modifier „transient“ ................................................................................................ 408 Nicht serialisierbare Basis-Klassen ....................................................................... 409 Aufgaben .............................................................................................................. 409 Lag. zu Aufgabe „Scribble 7“ – Kap. 25.6.1 .......................................................... 409 Abbildungen Abb. 2-1 : Executable-Abhängigkeit zu einer konkreten Plattform ........................................ 16 Abb. 2-2 : Executables für mehrere Plattformen ................................................................... 16 Abb. 2-3 : Executable für mehrere Plattformen entwickeln ................................................... 17 Abb. 2-4 : Idee einer virtuellen Maschine.............................................................................. 17 Abb. 2-5 : Portabilität mit einer VM und genormtem Byte-Code ........................................... 18 Abb. 3-1 : Oracle Homepage – Einstieg in den Download des JDK...................................... 32 Abb. 4-1 : Oracle Homepage – Einstieg in den Download des JDK...................................... 45 Abb. 4-2 : Oracle Java Download Seite – JDK Auswahl ....................................................... 46 Abb. 4-3 : Oracle JDK Download Seite – Plattform auswählen und downloaden .................. 47 Abb. 4-4 : Oracle Java Download Seite – JDK Dokumentation Auswahl .............................. 48 Abb. 4-5 : Oracle JDK Doku Download Seite – Doku auswählen und downloaden............... 48 Abb. 4-6 : Installiertes JDK – Aufruf der JVM ....................................................................... 49 Abb. 4-7 : Aufruf des Java-Compilers nach gesetzter Path Umgebungs-Variable ................ 49 Abb. 4-8 : Aufruf des Java-Compilers mit vollem Pfad ......................................................... 50 Abb. 4-9 : Entpackte Java Dokumentation ........................................................................... 50 Abb. 4-10 : Java Dokumentations Einstieg „index.html“ ....................................................... 51 Abb. 4-11 : Eclipse Homepage – Wechsel zur Download Seite............................................ 52 Abb. 4-12 : Eclipse Download-Seite – Eclipse IDE for Java Developers ............................... 53 Abb. 4-13 : „Hallo Welt“ Quelltext im Editor (Datei „e:\java\HalloWelt.java“) ......................... 55 Abb. 4-14 : Kommandozeile für das Verzeichnis „e:\java“ .................................................... 56 Abb. 4-15 : Der Java Compiler erzeugt den Byte-Code des „Hallo Welt“ Beispiels .............. 57 Abb. 4-16 : Die virtuelle Java Maschine (JVM) führt das „Hallo Welt“ Beispiel aus ............... 57 Abb. 4-17 : GUI Quelltext im Editor (Datei „e:\java\Gui.java“) ............................................... 58 Abb. 4-18 : Der Java Compiler erzeugt den Byte-Code des GUI Beispiels ........................... 58 Abb. 4-19 : Die virtuelle Java Maschine (JVM) führt das GUI Beispiel aus ........................... 59 Abb. 4-20 : Der Java Compiler erzeugt den Byte-Code aller Klassen im Verzeichnis ........... 59 Abb. 4-21 : Package Quelltext im Editor (Datei „PackageBeispiel.java“) .............................. 60 Abb. 4-22 : Verzeichnis-Struktur für das Package Beispiel................................................... 60 Abb. 4-23 : Der Java Compiler erzeugt den Byte-Code des Package Beispiels ................... 61 Abb. 4-24 : Fehler bei der Ausführung des Package Beispiels ............................................. 61 Abb. 4-25 : Der Java Compiler erzeugt den Byte-Code mit „sourcepath“ Option .................. 62 Abb. 4-26 : Korrekte Ausführung des Package Beispiels ..................................................... 63 Abb. 4-27 : Fehler bei Ausführung außerhalb des Wurzel-Verzeichnisses ........................... 63 Abb. 4-28 : Korrekte Ausführung des Package Beispiels mit Class-Path Option „-cp“.......... 64 Abb. 4-29 : Ausgangs Struktur für Byte-Code eigene Verzeichnisse .................................... 65 Abb. 4-30 : Der Java Compiler erzeugt den Byte-Code mit Trennung des Byte-Codes ........ 65 Abb. 4-31 : Erzeugte Verzeichnis Struktur und Byte-Code ................................................... 66 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 9 / 409 Abb. 4-32 : Korrekte Ausführung des Package Beispiels mit Trennung des Byte-Codes ..... 66 Abb. 4-33 : Kommandozeilen Argument Quelltext im Editor (Datei „e:\java\Args.java“) ........ 67 Abb. 4-34 : Compilation und Ausführung des Kommandozeilen Argumente Beispiels ......... 67 Abb. 4-35 : 2 Klassen in zwei parallelen Packages .............................................................. 69 Abb. 4-36 : Klassen, Dateien, Packages, Verzeichnisse und der Source-Path..................... 69 Abb. 4-37 : Eclipse Workspace Launcher ............................................................................ 71 Abb. 4-38 : Eclipse Begrüßungs Bildschirm.......................................................................... 72 Abb. 4-39 : Eclipse Begrüßungs Bildschirm schließen ......................................................... 72 Abb. 4-40 : Leerer Workspace in der Java Perspektive........................................................ 73 Abb. 4-41 : Neues Projekt anlegen....................................................................................... 74 Abb. 4-42 : Projekt-Wizard ................................................................................................... 75 Abb. 4-43 : Workspace mit neuem „Hallo Welt“ Projekt ....................................................... 76 Abb. 4-44 : Neue Klasse anlegen ......................................................................................... 77 Abb. 4-45 : Klassen-Wizard.................................................................................................. 78 Abb. 4-46 : Warnung bei Nutzung des Default-Packages .................................................... 79 Abb. 4-47 : Warnung bei fehlerhaftem Klassen-Namen ....................................................... 79 Abb. 4-48 : Neue Klasse „HalloWeltAppl“ im Workspace ..................................................... 80 Abb. 4-49 : Fertiger Quelltext im Wizard .............................................................................. 81 Abb. 4-50 : Starten des Programms ..................................................................................... 83 Abb. 4-51 : Ausgabe des Programms .................................................................................. 83 Abb. 4-52 : Eclipse Warnungs Preferences für „serialVersionUID“ ....................................... 85 Abb. 4-53 : Eclipse Warnungs Preferences für „typlose Container“ ...................................... 85 Abb. 4-54 : Ausgepackter Tutorial Workspace auf “E:” ........................................................ 86 Abb. 4-55 : Neuer Eclipse Workspace für die Tutorial Beispiele ........................................... 87 Abb. 4-56 : Kontext-Menü mit Import Eintrag ....................................................................... 87 Abb. 4-57 : Import Wizard Seite 1 ........................................................................................ 88 Abb. 4-58 : Import Wizard Seite 2 ........................................................................................ 89 Abb. 4-59 : Der fertig importierte Eclipse Workspace ........................................................... 90 Abb. 5-1 : Wert-Semantik ................................................................................................... 103 Abb. 5-2 : Referenz-Semantik ............................................................................................ 104 Abb. 9-1 : Ein String-Objekt wird nicht verändert ................................................................ 155 Abb. 9-2 : Ein StringBuilder-Objekt wird verändert ............................................................. 157 Abb. 9-3 : Warnungen des Java-Compilers wegen der Nutzung untypisierter Container.... 162 Abb. 9-4 : Warnungen der Eclipse 3.7.2 wegen der Nutzung untypisierter Container ........ 163 Abb. 9-5 : Vererbungs-Hierarchie der Wrapper-Klassen .................................................... 178 Abb. 10-1 : Referenz-Semantik bei Arrays ......................................................................... 207 Abb. 10-2 : Speicher-Layout eines mehrdimensionalen Arrays .......................................... 207 Abb. 10-3 : Speicher-Layout eines nicht-rechteckigen Arrays ............................................ 208 Abb. 11-1 : Das Schlüsselwort heißt „Abstraktion“ ............................................................. 210 Abb. 11-2 : Objekt-Kapselung und Benutzung über die Schnittstelle .................................. 210 Abb. 11-3 : Klassen und Objekte am Beispiel von Autos .................................................... 211 Abb. 11-4 : Funktionen ohne bzw. mit Objekt-Bezug .......................................................... 217 Abb. 12-1 : Türme von Hanoi ............................................................................................. 240 Abb. 14-1 : Vererbungs-Hierarchie ..................................................................................... 274 Abb. 14-2 : Allgemeine Vererbungs-Hierarchie .................................................................. 274 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 10 / 409 Abb. 14-3 : Vererbungs-Hierarchie des Beispiels ............................................................... 276 Abb. 14-4 : 2 Schicht-Architektur eines Programms ........................................................... 294 Abb. 14-5 : Abhängigkeit in einer 2 Schicht-Architektur ...................................................... 295 Abb. 14-6 : Design der Modul-Entkopplung ........................................................................ 295 Abb. 16-1 : Ein grafisches „Hallo Welt“ Programm ............................................................. 318 Abb. 16-2 : Ein GUI Programm mit bunter Ausgabe im Fenster ......................................... 321 Abb. 16-3 : GUI-Fenster aus Kapitel 19.4 mit Swing-Style ................................................. 322 Abb. 16-4 : GUI-Fenster aus Kapitel 19.4 mit Windows-Style ............................................ 323 Abb. 16-5 : Fenster mit schwebender Kugel ....................................................................... 324 Abb. 17-1 : Aufbau klassischer Programme ....................................................................... 326 Abb. 17-2 : Aufbau Event-gesteuerter Programme............................................................. 326 Abb. 18-1 : Der Fluß eines Events...................................................................................... 328 Abb. 18-2 : Ein einfaches Scribble Zeichen-Programm ...................................................... 331 Abb. 18-3 : Klassen-Diagramm Observer-Pattern .............................................................. 334 Abb. 18-4 : Interaktion Observer-Pattern ............................................................................ 334 Abb. 18-5 Klassen-Diagramm Observer-Pattern in Java ................................................... 334 Abb. 19-1 : Ein Fenster mit einem Button ........................................................................... 349 Abb. 19-2 : Die Content-Pane enthält nur den letzten Button ............................................. 350 Abb. 19-3 : Fenster mit Border-Layout ............................................................................... 351 Abb. 19-4 : Fenster mit Flow-Layout................................................................................... 351 Abb. 19-5 : Fenster mit Grid-Layout ................................................................................... 352 Abb. 19-6 : Fenster mit verschachtelten Layouts ................................................................ 353 Abb. 20-1 : Fenster mit Label ............................................................................................. 355 Abb. 20-2 : Fenster mit Text-Feld und automatischer Längen-Angabe .............................. 356 Abb. 20-3 : Fenster mit Button im initialen Zustand und nach dreimaliger Betätigung ........ 357 Abb. 20-4 : Fenster mit einer Gruppe von Radio-Buttons ................................................... 359 Abb. 20-5 : Fenster mit drei Check-Boxen .......................................................................... 360 Abb. 20-6 : Fenster mit einfacher 4x3 Tabelle .................................................................... 362 Abb. 20-7 : Fenster mit 40x30 Tabelle ohne Scrollbar ........................................................ 362 Abb. 20-8 : Fenster mit 40x30 Tabelle mit Scrollbar ........................................................... 363 Abb. 20-9 : Fenster mit Tabelle, dessen Inhalt aus einem Array stammt ............................ 364 Abb. 20-10 : Fenster mit Tabelle mit Tabellen-Modell ........................................................ 365 Abb. 20-11 : Fenster mit Menü ........................................................................................... 367 Abb. 20-12 : Fenster in grün und in rot – gesteuert von einem Timer ................................. 368 Abb. 22-1 : Ein kleiner aber wichtiger Teil der Exception-Hierarchie in Java ...................... 381 Abb. 22-2 : Klassen-Hierarchie für unsere Beispiel File-Exceptions ................................... 382 Abb. 23-1 : Ein nicht näher spezifizierter Datenstrom ......................................................... 393 Abb. 23-2 : Bsp für ineinander gekapselte Streams ........................................................... 394 Abb. 23-3 : Arbeitsweise der Stream-Kapselung ................................................................ 394 Versions-Historie Version Veränderte Kapitel © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 11 / 409 29 April 2016 • In vielen Kapiteln Rechtschreibfehler und kleine Ausdrucksfehler beseitigt. • Text in hoffentlich allen Kapiteln an die neuen Versionen von Java (JDK 1.8.0_u77) und Eclipse 4.5.2 (Mars) angepaßt – Screenshots sind noch auf dem alten Stand. • Viele Kapitel in Kapitel 6 über Operatoren überarbeitet – aber noch nicht fertig. • Einführendes Kapitel 3.9 über „Exceptions“ überarbeitet und leicht erweitert. • Einführendes Kapitel 3.10 über „Einlesen von der Kommandozeile“ aktualisiert. • Alle Quelltexte in den Kapiteln 8-11 mit Syntax-Highlighting versehen. • Kleine Änderungen in Kapitel 8. • Mehrere Fehler in den Beispielen von Kapitel 9 beseitigt, außerdem noch mehrfach in den Beispielen die Typ-Angabe bei Containern vereinfacht. • In den Musterlösungen zu Kapitel 11 auf typisierte Container umgestellt. • Kapitel 12.2.1 über Attribut-Initialisierungen erweitert • Kapitel 12.6 über Enums komplett neu geschrieben 28 April 2015 • In vielen Kapiteln Rechtschreibfehler und kleine Ausdrucksfehler beseitigt. • Text in hoffentlich allen Kapiteln an die neuen Versionen von Java (JDK 1.8.0_u40), Eclipse 4.4.2 (Luna) und NetBeans angepaßt. • Beschränkungen auf alte JDKs entfernt – das Tutorial geht jetzt defaultmäßig immer von einem JDK 1.8 aus. • Alle Quelltexte in den Kapiteln 1-7 mit Syntax-Highlighting versehen. • Viele kleine Erweiterungen, Anpassungen und Vereinfachungen im Kapitel 3. • Kapitel 4 in vielen Teilen erweitert, so z.B. ein neues Kapitel für den Import von Projekten in einen Eclipse-Workspace. Außerdem alle Abbildungen erneuert und an Java 8 und Eclipse 4.4 angepaßt. • Viele kleine Erweiterungen, Anpassungen und Vereinfachungen im Kapitel 5. 27 Mai 2012 • Text in hoffentlich allen Kapiteln an die neuen Versionen von Java (JDK 1.7.0_u04), Eclipse 3.7.2 und NetBeans (7.1.2) angepaßt. • In vielen Kapiteln Rechtschreibfehler und kleine Ausdrucksfehler beseitigt. • Kapitel 2.3.8 über JDK 1.7 vervollständigt. • Kleine Änderungen in den Schleifen-Kapiteln 7.3, 7.4 und 7.7.2. • Kleinere Ergänzungen in Kapitel 8.7 (Rekursion). • Kapitel 9 (Java-Bibliothek) in allen Kapiteln erweitert und überarbeitet – vor allem Kapitel 9.1 (Strings), den Anfang von Kapitel 9.3 (Container), Kapitel 9.5 (Zufallszahlen) und Kapitel 9.7 (Datei-System). Außerdem zwei neue Aufgaben 9.8.7 und 9.8.8 hinzugefügt – inkl. den Muster-Lösungen 9.15 und 9.16 (aber leider nur der reine Quelltext, noch ohne weitere Erklärungen). • Einige kleinere Änderungen in Kapitel 10 (u.a. neuer For-Schleifen-Typ), aber auch ein komplett neues Unter-Kapitel 10.3. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 12 / 409 26 Okt. 2011 • Text in vielen Kapiteln (aber noch nicht allen) an die neuen Versionen JDK 1.7.0_00 und Eclipse 3.7.1 angepaßt. • In vielen Kapiteln Rechtschreibfehler und kleine Ausdrucksfehler beseitigt. • Ein Unterkapitel 2.3.8 über JDK 1.7 in das Kapitel 2.3 aufgenommen, aber bisher nur angefangen – noch viel offen. • Kapitel 7.2 komplett neu geschrieben mit mehr Beispielen und auch JDK 1.7 Erweiterungen. • Kapitel 8 hat viele kleine Änderungen und Erweiterungen bekommen. • Kapitel 8 hat zwei neue Aufgaben (Kapitel 8.8.3 und Kapitel 8.8.4) mit Musterlösungen ohne Erklärungen (Kapitel 8.11 und Kapitel 8.12) bekommen. • Kapitel 9.3 vollkommen neu geschrieben mit vielen JDK 1.5 und JDK 1.7 Neuheiten und viel mehr Informationen über die Container-Klassen. Das Kapitel ist aber noch nicht fertig und enthält noch einige offene Punkte. • Kapitel 9.7 leider immer noch nicht neu geschrieben, aber immerhin schon Verweise auf einige „java.nio.file“ Neuigkeiten aus dem JDK 1.7 eingearbeitet. 25 Apr 2011 • Text in vielen Kapiteln an die neuen Versionen JDK 1.6.0_24 und Eclipse 3.6.2 angepaßt. • In vielen Kapiteln Rechtschreibfehler und kleine Ausdrucksfehler beseitigt. • Titelblatt eingeführt – mit Copyright und Hinweis auf die freie Verfügbarkeit des unveränderten Tutorials für Unterrichtszwecke. • Fußzeile um Link auf meine Homepage erweitert • Kapitel 1 um Copyright und Hinweis auf die freie Verfügbarkeit des unveränderten Tutorials für Unterrichtszwecke erweitert. • Kapitel 2.3.8 ist neu • Kapitel 16.6 über die Änderung des Swing GUI-Skins zu z.B. einem Windows-Style wurde neu aufgenommen. 24 • Kapitel 6.3 über die Geteilt- und Modulo-Operatoren ergänzt. • Fehler im ersten Beispiel in Kapitel 8.6 entfernt. • Fehler im Durchlauf-Zähler in der Lösung zur rekursiven Quadratwurzel-Näherung in Kapitel 8.13.2 entfernt. • Kapitel 9.1.2 über String-Verkettung erweitert. • Kapitel 9.2 um die Klasse „StringBuilder“ erweitert. • Sämtliche Lösungen des Kapitels 9 auf StringBuilder, typisierte Container und den neuen For-Schleifen-Typ umgestellt (Kapitel 9.10, 9.11, 9.12, 9.13 und 9.14). Außerdem Lösung für das Lottozahlen-Programm (Kapitel 9.13) optimiert. • Kleine Fehler in den Beispielen in Kapitel 11.4.2 beseitigt. 23 • Text in vielen Kapiteln an die neuen Versionen JDK 1.6.0_22 und Eclipse 3.6.1 angepaßt. • Kleine Fehler in vielen Kapiteln entfernt. • Break und Continue Anweisungen mit detailierten Beispielen hinterlegt – Kapitel 7.7. • Kapitel über echte Enums ergänzt – Kapitel 12.6. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Offene Punkte • • • • • • Seite 13 / 409 Thread.sleep aus Kapitel 11.3.1 in Kapitel 9 aufnehmen. Kapitel 7 ist in manchen Unterkapiteln noch ein bisschen knapp. Kapitel 18 hat noch einige offene Stellen. In Kapitel 23 fehlen noch einige Erklärungen. In Kapitel 25 fehlt der Hinweis auf das „serialVersionUID“ Attribut. Kapitel 10 über Arrays vor Kapitel 9 ziehen, da im File-System Kapitel ein Array von Strings vorkommt. 1 Organisatorisches Thema • Objektorientiertes Programmieren in Java • Achtung – die Sprache ist so umfangreich, dass die Vorlesung aus Zeitmangel bei weitem nicht alle Sprachelemente abdecken kann, und bei den abgedeckten auch viele Details ausläßt. • Achtung – die Bibliotheken von Java sind extrem umfangreich – auch hier wird die Vorlesung nur einzelne kleine Teile vorstellen können. Voraussetzungen • Kenntnisse einer beliebigen Programmiersprache. • Die Vorlesung setzt voraus, dass Sie die Grundlagen der Programmierung kennen. Sie sollten z.B. wissen was eine Variable, ein Typ oder eine Funktion ist, und warum und wofür man sie benutzt. Sie sollten die Begriffe Scope (Block), Lebensdauer, Speicher, Stack und Heap zumindest ungefähr zuordnen können. Über den Vorteil einer konsistenten und durchgängigen vorstellen Formatierung und Benamung sollte ich kein Wort mehr verlieren müssen. Ihnen sollte klar sein, warum die Auftrennung in möglichst unabhängige Module sehr sinnvoll ist. Und zu guter Letzt sollten Sie zumindest in Ansätzen in der Lage sein, ein Problem in Teil-Probleme zu zerlegen, es zu strukturieren, und einfache Algorithmische Lösungen zu verstehen und erarbeiten zu können. Tutorial • Das Tutorial wurde mal wieder überarbeitet und erweitert. Und trotzdem ist es leider noch nicht fertig – es enthält also noch immer einige Lücken und wahrscheinlich auch noch Fehler. Lesen Sie es also mit einer gesunden Skepsis – wie aber eigentlich alles, nicht wahr? • Das Tutorial bezieht sich prinzipiell auf die aktuelle Java 8 Plattform mit dem JDK 1.8. Aus Zeitmangel werden aber viele der „neueren“ Features von Java nicht besprochen. Von daher laufen viele Beispiele auch mit alten Java-Versionen wie dem JDK 1.5. Trotzdem setze ich bei allen Beispielen ein JDK 1.8 voraus – da ich zum Teil auch neue Sprach-Features vorstelle und nutzen werde. • Die Abbildungen und Beispiele der Kommandozeilen und der IDE in Kapitel 4 beziehen sich noch auf ältere Versionen von Java – konkret auf das JDK 1.5.0_06. Bzgl. der im Kapitel vorgestellten Themen gibt es aber keine Unterschiede zu den neusten Versionen, von daher © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 14 / 409 stimmen diese Beschreibungen und Abbildungen auch weiterhin problemlos • Die Abbildungen der GUI Programme wurden unter Windows XP als auch Windows 7 mit klassischem GUI Stil sowohl mit dem JDK 1.4.2, JDK 1.5, JDK 1.6, JDK 1.7 als auch dem JDK 1.8 erstellt. Je nach von Ihnen verwendetem Betriebssystem und/oder JDK kann der GUI Stil natürlich abweichen – verhalten sollten sich alle Beispiele aber identisch. • Sollten Sie Fehler finden oder Anregungen haben, so schicken Sie mir bitte eine Mail Adresse: ‘detlef@wilkening-online.de’. Bitte haben Sie Verständnis dafür, dass ich nicht jede Mail beantworten kann. Praktikum • Kleine und große Aufgaben in Java lösen • Näheres siehe in Kapitel 4. Tools • Die für die Entwicklung notwendigen Tools werden in Kapitel 4.1 vorgestellt. • Ihre Benutzung wird in den Kapiteln 4.3 und 4.6 kurz erläutert. Warum sollte ich Java lernen? • Im Augenblick die wohl wichtigste und verbreiteste Programmiersprache. • Relativ einfach zu lernen. • Im großen Maße plattform-unabhängig und weit verbreitet – siehe Kapitel 2.2. • Für viele Anwendungsfälle fertige Bibliotheken enthalten. • Gute Unterstützung an Tools – siehe auch Kapitel 4.1.1 und 4.1.3. • Gute Unterstützung an Bibliotheken, Literatur, usw. Literatur und WWW Es gibt hunderte von Büchern zu Java, Zeitschriften, und zig-Millionen Artikel im Web. Die erste Anlaufstelle sollte die offizielle Doku von Oracle sein: • http://www.oracle.com/us/technologies/java/index.html Ansonsten kann Sie z.B. Onkel „Google“ mit vielen weiteren Links und z.B. Tante „Amazon“ mit massenhaft Buch-Vorschlägen versorgen – auch wenn Sie natürlich andere Onkels und Tanten präferieren dürfen. 2 Java Dieses Kapitel führt in die Philosophie und Historie von Java ein, und stellt wichtige Schlagwörter und (Marketing-) Begriffe der Java Welt vor. Betrachtet werden z.B. die Implikationen und Vor- und Nachteile einer virtuellen Maschine, wie Java sie einsetzt oder die Historie der JDKs. Aber es wird eben auch erklärt, was eine virtuelle Maschine ist, was JDKs sind, oder was Applets oder Midlets sind. Alle, die dieses Hintergrund-Wissen nicht interessiert und die heiß auf die Sprache Java an sich und praktisches Programmieren sind, können direkt zum Kapitel 3 springen. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 15 / 409 2.1 Einleitung Java wurde von James Gosling bei Sun entwickelt. Die erste offizielle Vorstellung von Java war im März 1995. Java entwickelte sich in den kommenden Jahren – aus verschiedenen Gründen – zu einer der wichtigsten Programmier-Sprachen auf dem Markt, und ist seit einigen Jahren die wohl am häufigsten eingesetzteste Programmier-Sprache1 der Welt. Mittlerweile hat Oracle Sun übernommen – und damit liegt die Weiterentwicklung von Java primär in den Händen von Oracle. Bevor wir aber ab Kapitel 3 tiefer in die Sprache Java einsteigen, wollen wir einige andere Dinge klären, die Java von vielen anderen Sprachen unterscheidet, und zum Teil auch für den Erfolg von Java verantwortlich sind. So ist Java mehr als nur die objektorientierte Programmier-Sprache, als die Sie den Namen Java möglicherweise kennengelernt haben. Mit dem Begriff Java verbindet man auch häufig die sogenannte virtuelle Maschine „JVM“. Und manchmal wird Java sogar als Plattform – quasi als eine Art Betriebssystem – bezeichnet. Alle drei Sichtweisen haben ihre Berechtigung – und wir werden auch gleich sehen, warum. 2.2 Java als Plattform Um die Aussage „Java als Plattform“ zu verstehen, wollen wir uns etwas detailierter anschauen, wie „normale“ Compiler-Sprachen arbeiten (Kapitel 2.2.1), und dies dann mit Java vergleichen (Kapitel 2.2.2). 2.2.1 Compiler-Sprachen 2.2.1.1 Executable Bei einer Compiler-Sprache wie z.B. C oder C++ schreibt der Programmierer sein Programm in der entsprechenden Programmiersprache (eben z.B. C oder C++), und ein anderes Programm (der Compiler) übersetzt dies in direkt ausführbaren Maschinen-Code. Damit ist klar, dass das eigentlich Executable (d.h. die ausführbare Datei auf Ihrer Festplatte) Achtung – misstrauen Sie solchen Aussagen wie die häufigste, verbreiteste oder sonstwas Programmier-Sprache. Erstmal ist die Frage, wie definiert man sowas überhaupt (Anzahl an Code-Zeilen, Anzahl Programmierer, Anzahl Bibliotheken, ...)? Und letztlich kann keiner feststellen, wie viele Zeilen Code wirklich weltweit in irgendeiner Sprache geschrieben wurden, oder wieviele Programmierer oder Firmen eine Sprache einsetzen, oder sonstwas. Es gibt aber schon einige ganz interessante und halbwegs akzeptabler Statistiken über die Verbreitung und Relevanz von Programmier-Sprachen. Und hier liegt Java immer in der Spitzengruppe, häufig an erster Stelle. 1 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 16 / 409 fest an einen Prozessor (bzw. kompatible Prozessoren) gebunden ist, denn nur diese Prozessoren verstehen diesen Maschinen-Code. Z.B. ein Executable, compiliert für einen Sparc-Prozessor, läuft nicht auf einem Power-PC Prozessor – auch wenn das gleiche Betriebssystem vorhanden ist. Weiterhin hat jedes Betriebssystem ein spezielles Format, in dem ausführbare Datei aufgebaut sein müssen. Der Compiler muß beim Executable erzeugen natürlich dieses Format berücksichtigen. Damit ist ein Executable auch ein konkretes Betriebssystem (oder ein kompatibles) gebunden. Ein Executable, z.B. compiliert für Windows, läuft nicht unter Linux – auch wenn der gleiche Prozessor vorhanden ist. Unser Executable ist also an eine Plattform bestehend aus Prozessor und Betriebssystem gebunden. Abb. 2-1 : Executable-Abhängigkeit zu einer konkreten Plattform Natürlich kann man sein Programm auch für eine andere Plattform übersetzen, wenn man denn einen Compiler für diese Plattform hat, und das Programm plattform-unabhängig geschrieben ist (siehe Kapitel 2.2.1.2). Aber für jede gewünschte Ziel-Plattform (d.h. jede gewünschte Kombination aus Prozessor und Betriebssystem) muß ein eigenes Executable erstellt werden. Abb. 2-2 : Executables für mehrere Plattformen 2.2.1.2 Programmierung Aber nicht nur der Compiler bestimmt die Ziel-Plattform unserer Programm-Entwicklung. Auch die Programmierung selber kann hier bestimmte Randbedingungen festlegen. Ein Programm existiert ja nicht im luftleeren Raum, sondern soll später mit dem Computer und dem Benutzer interagieren. Dazu greift es z.B. auf das Datei-System zu, und macht Ein- und Ausgaben. Hier hat der Programmierer drei Möglichkeiten: 1. Er beschränkt sich auf die Sprach- und Bibliotheks-Elemente, die im Umfang seiner Programmier-Sprache enthalten sind. Unter C und C++ ist dies nur ein extrem kleiner Bereich, der von den ISO Standards abgedeckt ist – z.B. gehören grafische Oberflächen oder Netzwerk-Bereiche nicht dazu. - Der Programmierer ist sehr eingeschränkt in seinen Möglichkeiten. - Dafür hat er ein Programm geschrieben, das prinzipiell für viele Plattformen compiliert werden kann. 2. Der Programmierer greift direkt auf das Betriebssystem zu2. Dann hat er alle Möglichkeiten des Betriebssystems zur Verfügung – er kann also z.B. grafische Oberflächen implementieren, oder Netzwerk-Funktionalitäten nutzen. 2 Auf das sogenannte Betriebssystem API (Application Programming Interface). © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 17 / 409 - Der Programmier hat alle Möglichkeiten des Betriebssystems. - Das Programm ist an dieses eine Betriebssystem gebunden, also nicht portabel. 3. Letzlich kann der Programmierer auch fremde (3-party) Bibliotheken nutzen (kommerzielle oder freie), die betriebssystem-abhängige Funktionalität kapselt, und diese unter mehreren Betriebssystemen zur Verfügung stellt. - Der Programmierer muß entsprechende Bibliotheken für seinen Compiler finden, und er muß sie einsetzen können und dürfen3. - Das Programm funktioniert dann für alle Plattformen, für die die Bibliothek vorhanden ist. Abb. 2-3 : Executable für mehrere Plattformen entwickeln 2.2.1.3 Portabiliät Portabilität ist bei Compiler-Sprachen also ein komplexes Thema, da dort die Hardware, der Prozessor, das Betriebssystem und alle genutzten 3-party Bibliotheken eine Rolle spielen. 2.2.2 Java Virtual Machine – JVM Während Compiler-Sprachen also ein Executable erzeugen, dass an eine ganz konkrete Plattform (Hardware, Prozessor, Betriebssystem) gebunden ist, geht Java einen anderen Weg. Java compiliert die Programme für eine sogenannte virtuelle Maschine – die „Java Virtual Machine“ (JVM), d.h. eine Plattform, die es in Silizium gegossen gar nicht gibt4. Statt dessen wird die virtuelle Maschine in Software geschrieben, und simuliert die Java Plattform auf einer beliebigen anderen Plattform, z.B. Windows, Linux oder Mac OS-X. Abb. 2-4 : Idee einer virtuellen Maschine Damit macht sich Java in einem hohen Maße von den Portabilitäts-Problemen von CompilerSprachen unabhängig. Es gibt nur noch eine Plattform – die Java Virtual Machine – egal auf welcher echten Plattform das Programm hinterher laufen soll. Solange auf der echten Plattform eine JVM existiert, können Java Programme dort ablaufen. Die Bedingungen klingen vielleicht harmlos, sind es aber in der Praxis oft nicht. So haben z.B. Compiler zum Teil unterschiedliche Bibliotheks-Formate, die sich manchmal sogar zwischen Compiler-Versionen geändert haben. Oder eine Bibliothek ist zwar für den gesuchten Compiler vorhanden, aber mit anderen nicht-kompatiblen Optionen gebaut. Sind Sie jetzt von zwei Bibliotheken, die mit inkompatiblen Optionen erstellt wurden, abhängig, so haben Sie verloren – selber schon mal erlebt. Oder die Lizenz-Bedingung paßt nicht, oder die Bibliothek kostet zuviel, oder ist in der Firma nicht validiert, oder oder oder. 4 Okay, es könnte so eine Plattform in Silizium geben. Aber obwohl es schon Versuche in dieser Richtung gab, gibt es die JVM bis heute nur in Software. 3 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 18 / 409 Daher erzeugt ein Java Compiler auch keinen Maschinen-Code und kein Executable, sondern sogenannten Byte-Code. Und die Datei bzw. die Dateien, die dabei erzeugt werden, sind natürlich auch keine Executables, sondern Dateien, die eine JVM zur Ausführung benötigten. Dieser Ansatz einer virtuellen Maschine hat natürlich einige Konsequenzen – im positiven wie im negativen Sinne, die wir im folgenden diskutieren wollen. 2.2.2.1 Portabilität Fangen wir mit einem klaren Plus an – die Portablität. Java Byte-Code ist natürlich extrem portabel. Er enthält keinerlei Abhängigkeiten zu irgendeiner Hardware, zu irgendeinem Prozessor oder irgendeinem Betriebssystem. Auch 3-party Bibliotheken machen hier keine Probleme, da sie selber auch wieder als Java Byte-Code vorliegen. Solange auf der eigentlichen Plattform eine JVM existiert, kann der Java Byte-Code dort ausgeführt werden. Und dazu muß der Byte-Code auch nicht neu compiliert werden, denn es gibt keine bzgl. der Byte-Code Definition keinerlei Unterschiede zwischen den JVMs verschiedener echter Plattformen5. Das Ganze geht sogar soweit, dass man Byte-Code von verschiedenen Quellen problemlos mischen kann, und problemlos woanders laufen lassen kann. Z.B. könnte ein Java Projekt zum Teil unter Windows mit Eclipse entwickelt werden, ein anderer Teil unter Solaris mit dem JavaCompiler von Sun, und ein dritter Teil unter Linux mit einem beliebigen anderen Compiler. Der Byte-Code aller drei Teil-Projekte kann einfach zusammengebracht werden, und läuft dann z.B. auch auf Mac OS-X ohne jede Änderung. Denn der Byte-Code und die Schnittstelle der JVM sind genau normiert. Abb. 2-5 : Portabilität mit einer VM und genormtem Byte-Code Das ist Portablität in einer ganz anderen Dimension im Vergleich zu den herkömmlichen Compiler-Sprachen. 2.2.2.2 Sandbox Zusätzlich zu der Portablität ermöglicht der Ansatz einer VM auch noch andere Optionen. Da der Byte-Code nicht direkt auf der Hardware und dem Prozessor läuft, und er auch nicht direkt auf das Betriebssystem zugreifen kann, läßt sich ein Java Programm prinzipiell vollständig kontrollieren. Alle Aktionen des Programms müssen letztlich durch die VM ausgeführt werden. Wenn die VM aber sicherheits-kritische Aktionen wie z.B. den Zugriff auf das Dateisystem nicht zuläßt, kann ein Java-Programm hier nichts kaputt machen. Eine virtuelle Maschine kann also für ein Java-Programm wie ein Sandkasten („Sandbox“) sein, 5 Von den immer vorhandenen Bugs wollen wir mal absehen. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 19 / 409 in dem es sich austoben, aber auch keinen Blödsinn anrichten kann. Genau dies beherrscht die JVM auch. Im Kontext eines Browsers als Applet ausgeführte Java Programme (siehe Kapitel 2.2.2.3 und Kapitel 21) haben eingeschränkte Rechte. Nur über Zertifikate und explizite NutzerBestätigungen können Sie an mehr Rechte kommen – ansonsten müssen sie in ihrem Sandkasten bleiben. Hinweis – die Idee des Sandbox funktioniert natürlich nur, wenn die JVM Implementierung keine Fehler aufweist, durch die die Programme den Sandkasten verlassen können. 2.2.2.3 Browser-Anwendungen Ideal ist diese Portabilität und das Sandbox-Prinzip natürlich z.B. für Programme aus dem Internet, die in jedem Browser laufen können sollen. Denken Sie sich eine Aufgabe, für die einfache Text-Darstellung im Browser nicht reicht – hier wäre es schön, im Browser des Nutzers ein kleines Programm ablaufen lassen zu können. Dies ist mit Java möglich, da es für alle halbwegs wichtigen Browser Java-Plugins gibt, d.h. JVMs für den Browser. Ein Server kann also ein Java-Programm als Byte-Code an einen Browser ausliefern, ohne sich Gedanken um die Ziel-Plattform machen zu müssen (Portabilität), und Sie als Nutzer können so ein Programm bedenkenlos in Ihrem Browser ausführen, da es sich nur im Sandkasten der JVM mit eingeschränkten Rechten bewegen darf6. So ein Java-Programm, das im Browser läuft, ist übrigens kein ganz normales Java-Programm, sondern ein sogenanntes Applet, das sich minimal von normalen Java Desktop Programmen unterscheidet. Im Tutorial lernen wir Applets und ihre Programmierung in Kapitel 21 kennen. Hinweis – während in den Anfangszeiten von Java gerade die Applets als das Besondere von Java herausgestellt wurden und ein Grund für den Erfolg von Java waren – sind Applets heute nur noch sehr selten anzutreffen. Zum einen gibt es mittlerweile viele andere Technologien, die Java im Browser überflüssig machen, zum anderen hat sich die JVM im Browser als ein echtes Sicherheitsproblem herauskristallisiert – da die Sandbox nie wirklich sicher war. Die JVM hatte und hat immer mit schweren Lücken im Sandbox-Prinzip zu kämpfen, wodurch Applets den Sandkasten verlassen und dann mit den Rechten des Browsers auf dem System agieren konnten. 2.2.2.4 Portable Umgebungen Nicht nur Desktops bieten eine große Auswahl an Plattformen, auch portable Geräte wie z.B. PDAs oder Handies sind eine interessante Umgebung für portable Programme. Sie bieten vielfältige Hardware und unterschiedliche Betriebssystem – es wäre aber interessant Programme für alle diese Plattformen schreiben zu können. Dafür wurde in der Java Familie „JME“ („Java Mobile Edition“) entwickelt. Dies ist eine 6 Wir wollen wieder mal von Bugs in der JVM absehen. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 20 / 409 eingeschränkte JVM, die nicht alle Sprach-Features von Java anbietet, und auch bei weitem nicht alle Bibliotheken von Java unterstützt. Aber JME ist wieder eine genau definierte Umgebung mit klar abgesteckten Fähigkeiten, die an die Anforderungen von einfachen portablen Geräten angepaßt sind. Java-Programm für die JME-VM nennen sich Midlets. Achtung – Java Mobile ist schon einige Jahre alt und adressierte portable Umgebungen, die nicht besonders leistungsfähig waren – z.B. Handies aus einer Vor-Smartphone-Area. Heutige portable Geräte wie Smartphones oder Tablets haben die Leistungsfähigkeit von normalen Desktop Rechnern – hier kann eine fast ganz normale JVM problemlos laufen. 2.2.2.5 Android Viele heutige portable Geräte laufen unter dem Betriebssystem Android – und die Haus- und Hofsprache von Android ist Java. Wenn Sie eine App für Android entwickeln wollen, dann werden Sie dies in den meisten Fällen in Java machen. Natürlich sind auch andere Sprachen möglich – allen voran C++ – aber die normale Sprache für Android-Apps ist Java. Das Java von Android ist nicht abgespeckt wie JME, sondern ein ganz normales Java. Nur für die grafische Oberfläche wird nicht AWT, Swing oder JavaFX (siehe Kapitel 16) eingesetzt, sondern ein spezielles Android-Spezifisches GUI Framework. Leider sprengt die Entwicklung von Android Apps unseren Rahmen – obwohl es natürlich „cool“ wäre, ein Programm zu schreiben, das auf vielen Smartphones läuft. 2.2.2.6 Server-Umgebungen Was für Browser und PDAs recht ist, kann für Server-Infrastrukturen auch passend sein. Auch dort findet man häufig – historisch bedingt – ein heterogene und gewachsene Infrastruktur von Plattformen der verschiedensten Art. Früher konnten hierbei Programme nicht einfach von einem System auf das nächste verschoben werden, um den wechselnden Anforderungen zu genügen – mit Java ist dies möglich. Solange ein JVM zur Verfügung steht, kann ein Programm problemlos von einem Rechner auf den anderen verschoben oder kopiert werden. In Server-Umgebungen ist dies nicht das einzig interessante Kriterium. Wichtig ist z.B. auch die Unterstützung von Persistenz, Datenbanken, Transaktionen, uvm. Hierfür wurde in die Java Familie „Java EE“ („Java Enterprise Edition“) aufgenommen. Dies ist eine Erweiterung von Java um spezielle Bibliotheken und Fähigkeiten (z.B. „Java Enterprise Beans“ als ein KomponentenKonzept für Server-Anwendungen) für eben solche Themen. Auch andere Server-Themen fanden ihren Niederschlag in der Java Familie. So gibt es in Java z.B. Servlets – d.h. Java-Programme, die in einem Web-Server ausgeführt werden und dynamische HTML Seiten bereit stellen können. Oder auch Portlets, die speziell für die Programmierung von Web-Portalen entwickelt wurden. Im Tutorial wird gar nicht auf spezielle Server-Elemente von Java eingegangen – dazu fehlt © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 21 / 409 wieder mal die Zeit. 2.2.3 Plattform „Java“ Wie man sieht, stellt Java in Form der JVM wirklich eine Plattform dar – denn sie simuliert einen Java Prozessor mit einem Java Betriebssystem – dem typischen Verständnis einer Plattform. Aber auch mit einem breiter gefassten Verständnis von Plattform trifft der Begriff auf Java zu. Java adressiert neben den normalen Desktop-Programmen auch Applets, die im Browser laufen. Dann mit JME Midlets für einfache portable Geräte, und mit speziellen GUI Android Smartphones. Und für Server-Anwendungen gibt es Java EE mit u.a. Servlets und Portlets. Dies ist ein breites Spektrum an Einsatzgebieten, was in dieser Vielfalt von kaum einer anderen Programmier-Sprache abgedeckt wird. 2.3 Java Versionen Als Java im März 1995 der Öffentlichkeit vorgestellt wurde, war Java noch eine ziemlich kleine Sprache – und kaum jemand (niemand?) dachte an eine solche Verbreitung, wie sie heute vorhanden ist. Die initiale Entwicklung von Java hatte auch nur wenig mit den heutigen Einsatzgebieten zu tun, sondern es ging um eine Software-Plattform für Haushaltsgeräte7. Da Haushaltsgeräte auch eine sehr hetorogene Plattform darstellen, war die Entwicklung einer virtuellen Maschine natürlich eine gute Idee8. Wenn Sie Java nur als Ausführungs-Plattform nutzen wollen – und das wird auf die meisten Menschen zutreffen – dann benötigen Sie nur die JVM in Form des JRE. Wollen Sie auch mit Java entwickeln – und das betrifft uns – dann benötigen wir das JDK. JRE steht hierbei für „Java Runtime Environment“ und ist das Stück Software, was der Nutzer braucht, um ein Java-Programm (d.h. den Java Byte-Code) laufen zu lassen – also im Prinzip die virtuelle Maschine. JDK steht für „Java Development Kit“ und enthält neben dem JRE auch die Tools und Bibliotheken, die man braucht um Java Programme zu entwickeln. Ein JDK ist also das, was Sie zum Entwickeln brauchen – siehe auch Kapitel 4.1.1. Häufig wird es auch als Java SDK („Software Development Kit“) bezeichnet. Sie kennen doch sicher auch diese Visionen vom Kühlschrank, der merkt, wenn die Milch alle ist und beim Händler direkt neue bestellt. Hier hat auch der vielzitierte Satz „Java wurde für Toaster entwickelt“ seinen wahren Ursprung. 8 Aber glauben Sie jetzt nicht, dass die Idee einer VM eine Java Idee ist. Schon Anfang der 70er Jahre gab es erste VMs mit normiertem Byte-Code, z.B. Eumel. Aber damals war die Zeit noch nicht reif dafür. 7 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 22 / 409 2.3.1 JDK 1.0 Aus dem Labor kam Java heraus, als Sun eine Konkurrenz zu Microsofts Vorherrschaft auf dem Desktop suchte, und das Internet sich zu entwickeln began. Der Desktop sollte unwichtig werden, und statt dessen sollte der Nutzer einfach alles im Browser und dem Internet machen. Und hierfür erschien eine VM mit ihren Fähigkeiten bzgl. Portabliltät und Sicherheit natürlich ideal. Also wurde die Idee von Applets entwickelt – die Möglichkeit das damals recht beschränkte Web um interaktive Möglichkeiten zu erweitern. So kam Java mit dem JRE 1.0 bzw. dem JDK 1.0 in die Welt. Applets haben aber nicht wirklich die Welt gerettet – oder wie häufig begegnen Ihnen Applets im Internet? Heutzutage gibt es andere Technologien um Rich-Internet-Seiten zu programmieren, die sich mehr verbreitet haben, so z.B. DHTML, JavaScript, Ajax oder Flash. Applets sind die Ausnahme, die man nutzt, wenn man wirklich die Vorteile einer mächtigen Programmiersprache und der Sandbox im Browser benötigt. Die Java Bibliothek des JDK 1.0 bestand aus 212 Klassen in 8 Packages. Selbst wenn Ihnen die Begriffe Klassen und Packages noch nichts sagen – wir lernen sie in Kapitel 11 bzw. 13 kennen – sind die reinen Mengen-Zahlen im Laufe der JDK-Historie ganz interessant. Im Augenblick reicht uns hier die einfache Vorstellung, dass Klassen stark vereinfacht (und auch nicht richtig) eine Bündelung von Funktionen darstellen, und Packages ihrerseits wieder Klassen in Gruppen strukturieren. Interessant ist also vor allem die Zahl der Klassen, da sie indirekt ungefähr die Zahl der Bibliotheks-Funktionen widerspiegelt. 2.3.2 JDK 1.1 Die nächste Java Version (JDK 1.1) wurde im Februar 1997 freigegeben. Diese Version war ein erster wichtiger Schritt zu einer ernsthaften Plattform. Der Sprung von Java aus dem Labor in die reale Welt war mit dem JDK 1.0 relativ schnell erfolgt, und so krankte Java noch an einigen Kinderkrankheiten. Das JDK 1.1 brachte hier viele Abhilfen, und so wurde neben der Bibliothek auch die Sprache erweitert. Mit dem JDK 1.1 gab es nun z.B. innere Klassen, das GUI EventModell wurde quasi auf den Kopf gestellt, es gab Änderungen an der Serialisierung und an Reflection, Unicode-Dateien wurden nun unterstützt, JDBC hielt Einzug, und die JNI Schnittstelle wurde standardisiert. Und die Bibliothek erweitert sich um fast 300 Klassen. JDK Version 1.0 1.1 Datum Packages Apr. 1995 8 Feb. 1997 23 Klassen 212 504 Achtung – nehmen Sie die Anzahl der Packages und Klassen nicht zu genau. Die Zählungen stellen Momentaufnahmen dar. Es ist immer wieder passiert, dass neuere Unterversionen des JDK (z.B. 1.4.2) neue Klassen und Packages eingeführt haben. 2.3.3 JDK 1.2 – Java 2 Zwischen den JDK Versionen 1.1 und 1.2 gab es viele Zwischen-Versionen, die manche © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 23 / 409 Neuerungen inkrementell einführten. Vor allem die GUI Welt von Java war im Umbruch – über das eher einfache AWT von JDK 1.0 und JDK 1.1 wurde das schönere und mächtigere Swing gesetzt. Damit wechselte das primäre Ziel von Java von den Applets zu Desktop-Programmen. AWT („Abstract Window Toolkit“) war der Teil der Bibliothek des JDK 1.0 und 1.1, welches für die GUI Programmierung zuständig war. AWT ist auch heute noch Teil von Java, wurde aber durch Swing erweitert, was eine viel leistungsfähigere GUI Bibliothek darstellt. Nähere Informationen zu AWT und Swing finden Sie in bzw. ab Kapitel 16. Eine weitere wichtige Änderung war eine starke Erweiterung der Collections – wichtig, da Collections das tägliche Brot des Programmierers darstellen, siehe auch Kapitel 9.3. Aber auch viele andere Bibliotheksteile wurden überarbeitet, erweitert oder ergänzt. So drückte sich das dann in Zahlen aus: JDK Version 1.0 1.1 1.2 (Java 2) Datum Apr. 1995 Feb. 1997 Nov. 1998 Packages 8 23 59 Klassen 212 504 1520 Achtung – nehmen Sie die Anzahl der Packages und Klassen nicht zu genau. Die Zählungen stellen Momentaufnahmen dar. Es ist immer wieder passiert, dass neuere Unterversionen des JDK (z.B. 1.4.2) neue Klassen und Packages eingeführt haben. Alles zusammen war Sun Grund genug, von einem ganz neuen besseren Java zu reden, und gab Java mit dem Namen „Java 2“ die Versions-Nummer 2. Zusätzlich deutete sich mit dem JDK 1.2 schon ein Wechsel bzgl. der VM Technologie an. Seit dem JDK 1.3 ist HotSpot die primäre JVM von Sun (siehe Kapitel 2.4.2), im JDK 1.2 war sie schon enthalten und konnte per Option aktiviert werden. 2.3.4 JDK 1.3 Mit dem JDK 1.3 wurde der Focus von Java auch auf die Server-Infrastrukturen gelenkt, und so brachte das JDK 1.3 neben HotSpot (siehe Kapitel 2.4.2) viele im Server-Umfeld wichtige neue und erweiterte Bibliotheksteile mit – z.B. Verteilung via RMI/IIOP und JNDI Namensdienste. JDK Version 1.0 1.1 1.2 (Java 2) 1.3 Datum Apr. 1995 Feb. 1997 Nov. 1998 Mai 2000 Packages 8 23 59 76 Klassen 212 504 1520 1842 Achtung – nehmen Sie die Anzahl der Packages und Klassen nicht zu genau. Die Zählungen stellen Momentaufnahmen dar. Es ist immer wieder passiert, dass neuere Unterversionen des © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 24 / 409 JDK (z.B. 1.4.2) neue Klassen und Packages eingeführt haben. 2.3.5 JDK 1.4 JDK 1.4 im Februar 2002 war eine für Java wichtige Version. Viele Kritiker sagen, dass dies die erste professionell und produktiv einsetzbare Version war. Auf jeden Fall zeichnete sich das JDK 1.4 durch weiter verbesserte Performance, einen besseren Garbage-Collector (siehe Kapitel 2.4.1) und hohe Stabilität aus. Auch heute wird das JDK 1.4 von Sun noch unterstützt, und wird auch noch von vielen Firmen aus den verschiedensten Gründen eingesetzt. Natürlich brachte auch das JDK 1.4 wieder viele Bibliotheks Verbesserungen und Erweiterungen mit sich – schauen Sie sich mal die Zahlen in der folgenden Tabelle an. Sowohl Desktop-Programme profitierten von neuen Swing Features, als auch Server-Programme von vielen anderen Neuerungen. Und mit JDK 1.4 wurde zum ersten Mal seit dem JDK 1.1 auch wieder die Sprache erweitert – Java bekam Assertions, die aus Zeitmangel in der Vorlesung leider nicht besprochen werden können. JDK Version 1.0 1.1 1.2 (Java 2) 1.3 1.4 Datum Apr. 1995 Feb. 1997 Nov. 1998 Mai 2000 Feb. 2002 Packages 8 23 59 76 135 Klassen 212 504 1520 1842 2723 Achtung – nehmen Sie die Anzahl der Packages und Klassen nicht zu genau. Die Zählungen stellen Momentaufnahmen dar. Es ist immer wieder passiert, dass neuere Unterversionen des JDK (z.B. 1.4.2) neue Klassen und Packages eingeführt haben. 2.3.6 JDK 1.5 – Java 5 Mit dem JDK 1.5 wurde Java mal wieder umbenannt: „Java 5“ hieß es nun. Obwohl rund 2 ½ Jahre seit dem JDK 1.4 ins Land gegangen waren, vergrößerte sich die Bibliothek kaum. Es kamen nur 556 neue Klassen hinzu. JDK Version 1.0 1.1 1.2 (Java 2) 1.3 1.4 1.5 (Java 5) © Detlef Wilkening 1997-2016 Datum Apr. 1995 Feb. 1997 Nov. 1998 Mai 2000 Feb. 2002 Sep. 2004 Packages 8 23 59 76 135 166 Klassen 212 504 1520 1842 2723 3279 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 25 / 409 Achtung – nehmen Sie die Anzahl der Packages und Klassen nicht zu genau. Die Zählungen stellen Momentaufnahmen dar. Es ist immer wieder passiert, dass neuere Unterversionen des JDK (z.B. 1.4.2) neue Klassen und Packages eingeführt haben. Die größten Änderungen passierten mit der Sprache selber und unter der Haube. Am offensichtlichsten waren hiervon natürlich die Sprach-Erweiterungen: • Neue Schleifen-Konstrukte • Aufzählungs-Typen • Generics (typisierte Container) • Annotations • statische Imports • etc. Unter der Haube beherrscht Java seit dem JDK 1.5 z.B. parallele Garbage-Collection – siehe Kapitel 2.4.1. Hinweis – aus Zeitmangel werden viele der mit Java 5 eingeführten Sprach-Konstrukte im Tutorial nicht oder nur sehr knapp besprochen. 2.3.7 JDK 1.6 – Java 6 Im Dezember 2006 beglückte Sun die Java Community mit Java 6 bzw. dem JDK 1.6 – der zur Zeit aktuellen Version9. Es gab keine Sprach-Änderungen, aber natürlich wurde die Bibliothek wieder verbessert und erweitert – so wurde u.a. die Integration von Java Programmen in den Desktop stark verbessert, z.B. durch eine System-Tray Unterstützung. Aber die vielleicht interessanteste Neuerungen spielte sich woanders ab. Seit dem JDK 1.6 hat Java eine Script-API und unterstützt damit direkt mehrere andere Programmierspachen. So gibt es jetzt z.B. JRuby, eine in Java geschriebene Implementierung der Programmiersprache Ruby auf der JVM mit der Möglichkeit der einfachen Interaktion zwischen beiden Welten. JDK Version 1.0 1.1 1.2 (Java 2) 1.3 1.4 1.5 (Java 5) 1.6 (Java 6) Datum Apr. 1995 Feb. 1997 Nov. 1998 Mai 2000 Feb. 2002 Sep. 2004 Dez. 2006 Packages 8 23 59 76 135 166 202 Klassen 212 504 1520 1842 2723 3279 3777 Achtung – nehmen Sie die Anzahl der Packages und Klassen nicht zu genau. Die Zählungen 9 Stand 13.04.2008 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 26 / 409 stellen Momentaufnahmen dar. Es ist immer wieder passiert, dass neuere Unterversionen des JDK (z.B. 1.4.2) neue Klassen und Packages eingeführt haben. 2.3.8 JDK 1.7 Mittlerweile hat Oracle Sun übernommen, und Java steht nun unter der Regie von Oracle. Dies hat dazu geführt, dass wir lange auf die nächste Java Version warten mußten. Aber seit August 2011 ist Java 7 (JDK 1.7) da, und seit Mai 2012 ist sie auch für den Produktiveinsatz freigegeben. Java 7 brachte sowohl Erweiterungen der Sprache, der Bibliothek als auch der JVM mit sich. Die Sprache Java wurde um einige kleine Dinge ergänzt, die das Leben in der Praxis einiges leichter machen. Zum Beispiel können jetzt Switch-Anweisungen für Strings genutzt werden (Kapitel 7.2), lesbare Integer- und Binär-Literale wurden eingeführt, Exception-Catch-Blöcke können nun mehrere Exceptions gleichzeitig fangen, Redundante Typ-Informationen bei Generics sind nicht mehr notwendig (Kapitel 9.3.1), oder die Handhabung von ClosableRessourcen ist nun viel sicherer. Auch in der Bibliothek gab es eine Menge neue Dinge – besonders hervorzuheben sind sicher das neue Fork/Join-Framework im Multithreading-Umfeld, die Erweiterungen im New-IO Package (Kapitel 9.7.2), Unterstützung von Unicode 6, Veränderungen an den Währung-Codes und Locales, Überarbeitung des XML-Stacks, und der Ausbau der GUI Bibliothek mit z.B. neuen Designs (Nimbus-Projekt) und transparenten bzw. halb-transparenten Fenstern. Unter der Haube, daher in der virtuellen Maschine, wurde das Class-Loading Verhalten verändert, der neue Garbage Collector G1 aktiviert, und die Anbindung an neue ProgrammierSprachen weiter vereinfacht. Unterm Strich finden sich vielleicht keine großen Erweiterungen in Java 7 wieder, aber doch viele kleine Dinge die das Programmierer-Leben einfacher und/oder sicherer machen. JDK Version 1.0 1.1 1.2 (Java 2) 1.3 1.4 1.5 (Java 5) 1.6 (Java 6) 1.7 (Java 7) Datum Apr. 1995 Feb. 1997 Nov. 1998 Mai 2000 Feb. 2002 Sep. 2004 Dez. 2006 Aug. 2011 Packages 8 23 59 76 135 166 202 209 Klassen 212 504 1520 1842 2723 3279 3777 4024 Achtung – nehmen Sie die Anzahl der Packages und Klassen nicht zu genau. Die Zählungen stellen Momentaufnahmen dar. Es ist immer wieder passiert, dass neuere Unterversionen des JDK (z.B. 1.4.2) neue Klassen und Packages eingeführt haben. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 27 / 409 2.3.9 JDK 1.8 – Java 8 Am 18. März 2014 ist Java 8 erschienen. Java 8 brachte sowohl Erweiterungen der Sprache, der Bibliothek als auch der JVM mit sich: in die Sprache wurden Lambdas, funktionale Interfaces, Default-Implementierungen und Streams mit parallelen Algorithmen eingeführt. Die Bibliothek bekam eine vollkommen neue Date-Time-API, viele Concurrency Erweiterungen, und Swing als GUI-Anteil wurde durch Java FX abgelöst. Unterm Strich sind die Erweiterungen in Java 8 so umfangreich, dass sie höchstens mit dem Sprung auf Java 5 vergleichbar sind. JDK Version 1.0 1.1 1.2 (Java 2) 1.3 1.4 1.5 (Java 5) 1.6 (Java 6) 1.7 (Java 7) 1.8 (Java 8) Datum Apr. 1995 Feb. 1997 Nov. 1998 Mai 2000 Feb. 2002 Sep. 2004 Dez. 2006 Aug. 2011 März 2014 Packages 8 23 59 76 135 166 202 209 217 Klassen 212 504 1520 1842 2723 3279 3777 4024 4240 Achtung – nehmen Sie die Anzahl der Packages und Klassen nicht zu genau. Die Zählungen stellen Momentaufnahmen dar. Es ist immer wieder passiert, dass neuere Unterversionen des JDK (z.B. 1.4.2) neue Klassen und Packages eingeführt haben. Hinweis – aus Zeitmangel werden viele der mit Java 8 eingeführten Sprach-Konstrukte im Tutorial nicht oder nur sehr knapp besprochen. 2.4 Technologien Die JVM enthält einige sehr interessante Technologien, die heutzutage die Leistungsfähigkeit von Java mitbestimmen und daher zum Erfolg von Java beigetragen haben. Zwei dieser Technologien möchte ich hier detailierter vorstellen, da sie im bisherigen Kapitel schon mehrfach erwähnt wurden, und sehr aufschlußreich sind. • Speicherverwaltung und Garbage-Collector • Performance und Hotspot 2.4.1 Speicherverwaltung und Garbage-Collector Die Sprache Jave und die JVM unterstützen eine Technik, die Garbage-Collection genannt wird. Dies meint einfach nur, dass Sie beliebig Objekte erzeugen können, und sich nicht um die Freigabe des von den Objekten belegten Speichers kümmern müssen. Dies macht dann der Garbage-Collector – ein Teil der JVM. Er erkennt nicht mehr benötigte Objekte und gibt deren Speicher wieder frei. Dies entlastet Sie als Programmierer, da Sie sich um dieses Thema nicht mehr kümmern müssen – und es verhindert den Quell von Fehlern der mit manueller Speicherverwaltung verbunden ist. Falls Sie jemals z.B. „C“ programmiert haben, dann wissen © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 28 / 409 Sie sicher, dass Zeiger und manuelle Speicherverwaltung eine Büchse der Pandorra sind, deren Problemen man kaum Herr wird. All das gibt es in Java nicht – das macht die Sprache einfacher und die Programme fehlerfreier. Eigentlich könnte man hier aufhören, denn damit ist das Thema „Speicherverwaltung und Garbage-Collection“ oberflächlich ausreichend beschrieben. Und uns fehlt die Zeit detailiert in die heutigen Techniken und Algorithmen von Garbage-Collectoren einzusteigen, obwohl es ein ungeheuer interessantes und faszinierendes Thema ist. Aber ein paar Dinge möchte ich Ihnen doch noch mit auf den Weg geben – heutige Garbage-Collectoren wie z.B. im JDK 1.8 arbeiten ungeheuer effizient, d.h. : • Sie erkennen zuverlässig nicht mehr referenzierte Objekte. • Sie arbeiten sehr schnell. Programme mit Garbage-Collections können den gleichen Programmen mit manueller Speicherverwaltung in der Performance überlegen sein. • Seit dem JDK 1.5 kann ein Großteil der Garbage-Collector Techniken auch parallel zum normalen Programmfluß stattfinden – daher die JVM nutzt z.B. neuere Multi-Core Maschinen sehr gut aus. Einziger Nachteil heutiger Garbage-Collector Techniken ist, dass sie einen gewissen Speicheroverhead haben, d.h. mehr Speicher verbrauen als im Idealfall notwendig wäre. Bei reinen Mark&Copy Garbage-Collectoren10 kann dieser Overhead fast die Hälfe des Speichers ausmachen – aber auch hier sind heutige JVM’s viel effizienter geworden. 2.4.2 Performance und HotSpot Ein immer wieder heißes Thema in der Diskussion um Java ist die Performance. Es ist ja auch scheinbar sehr einleuchtend, dass Byte-Code auf einem in Software geschriebenen Prozessor nicht so schnell ausgeführt werden kann wie Maschinen-Code, der direkt auf einem echten Prozessor ausgeführt wird. Aber das ist nur der erste Eindruck – mittlerweile gibt es über 40 Jahre Erfahrung und Ideen im Bau von virtuellen Maschinen, und viele dieser Ideen betreffen der Performance. Die beiden interessantesten Technologien sind hier JIT-Compiling und HotSpot. JIT steht für „Just-in-Time“, d.h. ein JIT-Compiler ist ein Just-in-Time Compiler. Damit ist gemeint, dass die virtuelle Maschine statt einfach den Byte-Code auszuführen diesen bei der ersten Ausführung in Maschinen-Code übersetzt und dann dieser direkt vom Prozessor ausgeführt wird. Da dieser Übersetzungs-Vorgang Zeit dauert, wird die Ausführungs-Dauer beim Übersetzen natürlich stark gebremst – danach steht aber die native ProzessorPerformance zur Verfügung. Eine Garbage-Collector Technik, die mit zwei Heaps arbeitet, von denen einer immer leer ist – d.h. sinnlos verbrauchter Speicher darstellt. Dafür ist die Technik sehr schnell, da sie noch referenzierte Objekte nur markieren („mark“) und dann umkopieren („copy“) muß, und die Speicherfreigabe der restlichen Objekte in Nullzeit erledigt, da sie sie einfach auf dem alten Heap vergißt. 10 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 29 / 409 Prinzipiell ließe sich natürlich der gesamte Java Byte-Code in Maschinen-Code übersetzen, aber es gibt Bereiche, wo es sich nicht lohnt, der Aufwand sehr hoch ist, oder aus anderen Gründen nicht wirklich gut praktikabel ist. Außerdem darf auch compilierter Byte-Code nicht die Sicherheits-Prinzipien von Java wie z.B. die Sandbox verletzen. Auf der anderen Seite gibt es aber auch Bereiche wo sich mit Informationen aus dem Kontext des Byte-Codes sehr radikale Optimierungen machen liessen – z.B. wer ruft eine Funktion auf, bzw. mit welchen Parametern wird sie immer aufgerufen. Und hier kommt HotSpot zum Tragen. HotSpot ist der Name der JVM seit dem JDK 1.3. Bestandteil von HotSpot ist u.a. ein JITCompiler. Das Besondere an HotSpot ist aber, dass hier die VM das Programm zur Laufzeit beobachtet, und in Abhängigkeit von diesen Beobachtungen weitere Optimierungen durchführt. Ein offensichtlicher Hintergrund dieser Vorgehensweise ist die Erkennung sogenannter HotSpots, daher von Programm-Teilen, die besonders häufig ausgeführt werden und bei denen sich die Optimierung besonders lohnt. Aber HotSpot kann z.B. auch dynamische Optimierungen durchführen, die die Semantik des Codes verändern, aber im konkret vorliegenden Kontext funktionieren11. Hinter HotSpot steht mittlerweile eine Menge Erfahrung und Know-How, und die Optimierungen werden mit jedem JDK verbessert. Mit JIT-Compiling und den weiteren HotSpot Optimierungen spielt Java heute von der Performance her auf dem Niveau von normalen compilierten Sprachen wie z.B. C++ mit. Es gibt sogar ernst zu nehmende Stimmen, die die Meinung vertreten, dass virtuelle Maschinen viel besser optimieren können als jeder Compiler, da ihnen viel mehr Informationen zur Verfügung stehen kann. 2.4.2.1 Bemerkung Eigentlich ist hiermit alles Wichtige gesagt. Natürlich ließe sich noch manches zu den Techniken von VMs, JIT-Compilern und vor allem HotSpot schreiben – aber eine solche Vertiefung würde den Rahmen des Tutorials bei weitem sprengen. Ich möchte aber noch ein bisschen zum Thema Performance von Java sagen, da es scheinbar ein extrem emotional besetztes Thema mit vielen Gerüchten und Halb-Wahrheiten ist. Viele Kritiker von Java haben bis heute nicht erkannt bzw. erkennen wollen, dass sich die Performance von Java seit dem JDK 1.0 immens verbessert hat, und viele ihrer Kritiken heute nicht mehr zutreffend sind. Auf der anderen Seite gibt es auch eine Menge Java-Fanatiker, die unreflektiert Java für die performanteste Sprache der Welt halten. Und wenn diese aufeinander treffen, dann knallt es – manchmal kann das sehr unterhaltsam sein. Ich kann Ihnen nur empfehlen, wenn Sie mal Langeweile haben – suchen Sie sich z.B. eine C++ Newsgroup und posten Sie etwas polemisch hinein, dass Java viel schneller ist als C++. Oder posten Sie umgekehrt in eine Java Newsgroup dass C++ viel schneller als Java ist. Dann lehnen Sie sich zurüch und genießen den Sprachen-Krieg, den Sie angezettelt haben. An 11 Man nennt solche Optimierungen „Closed-World Optimierungen“. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 30 / 409 langen verregneten Abenden kann dies viel unterhaltsamer als jeder Film sein. Und wo liegt die Wahrheit? Nirgendwo oder überall. Ich kann Ihnen problemlos ein Programm schreiben, wo C++ Java schlägt, und auch umgekehrt. Aber was sagt das schon aus? Nichts! Es kann schon unglaublich komplex und schwierig sein, „objektiv“ und reproduzierbar die unterschiedliche Performance zweier unterschiedlicher Implementierungen der gleichen Sprache in der gleichen Umgebung zu beurteilen. Viel schwieriger ist es schon, verschiedene Bibliotheken zu vergleichen. Wie schwierig ist es dann wohl, akzeptable Vergleiche zwischen zwei Sprachen zu bekommen? Und ist das überhaupt sinnvoll? Man entscheidet sich doch nicht für oder gegen eine Sprache nur aufgrund der Performance, sondern auch aufgrund von hunderten anderer Kriterien. Ich kann Ihnen nur empfehlen: solange Sie nicht wirklich richtig Ahnung von der Materie haben – halten Sie sich aus der Performance Diskussion raus. Java ist heute, gerade auf heutigen Rechnern, verdammt schnell – aber das gilt auch für viele andere Sprachen. Und wenn Sie wirklich mal ein Performance-Problem haben, suchen Sie nach besseren Algorithmen oder anderen Implementierungen bevor Sie die Sprache wechseln. In fast allen Fällen reicht das aus. Und wenn das nicht reicht, dann haben Sie wahrscheinlich ein Problem, das ein einfacher Sprachenwechsel auch nicht so einfach beseitigt. 2.5 Fazit Wie Sie sehen, ist gerade die Plattform Java in Form der JVM ein faszinierender Ansatz, der viele Probleme z.B. bzgl. Portabilität oder Speicherverwaltung radikal beseitigt. Hinzu kommt eine relativ einfache Sprache, mit der sich das restliche Tutorial beschäftigt, und eine wunderbare riesige Bibliothek, die viele Probleme löst. In Kapitel 4.1.3 werden wir dann noch einige sehr mächtige Tools kennen lernen, die uns beim Programmieren stark unterstützen und auch noch frei sind. Programmierer-Herz – was willst du mehr? Ach ja, es sollte Ihnen klar sein, dass dieses Kapitel bei weitem nicht alle Java Wörter vorgestellt hat. Wundern Sie sich also nicht, wenn Sie mit welchen konfrontiert werden, die hier fehlen. Das Java Universum ist groß und wächst jeden Tag – es gibt viel zu entdecken... 3 Mini-Einführung In Java können als ausführbare Einheiten Programme, Applets, Midlets und Servlets erstellt werden. Wir werden hier hauptsächlich Programme erstellen. Der Unterschied zwischen Applets und Anwendungen ist nicht sehr groß, so dass dies im Augenblick keine praktische Bedeutung hat12. 12 Es gibt Unterschiede in der Art des Einsprungs, der Sicherheit, der Bibliotheken und der Konsole - siehe todo... © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 31 / 409 Dieses Kapitel stellt einige Grundlagen vor, mit denen Sie ruck-zuck konfrontiert werden, um überhaupt kleine sinnvolle Programme schreiben zu können, und um die Praktikums-Aufgaben lösen zu können. Von daher brauchen wir sie, und darum werden sie hier gleich erklärt. Auf der anderen Seite sind viele dieser Dinge „fortgeschrittenen Themen“, die erst im Laufe der Zeit kommen. Nehmen Sie sie darum hier einfach hin, und wenden Sie sie pragmatisch an, ohne zuviel darüber nachzudenken. Im Laufe der Zeit klärt sich alles auf. 3.1 Applikation 3.1.1 Beispiel 1 In einer Java Applikation muss es mindestens eine public-Klasse geben, die folgende KlassenFunktion main enthält: public class Kap_03_01_Bsp_01_HalloWelt { public static void main(String[] args) { System.out.println("Hallo Welt"); } } Jede öffentliche Klassen-Funktion („public static“) mit dem Namen „main“, dem Rückgabetyp „void“, und der Parameterliste „String[ ]“ stellt einen Einsprungpunkt in ein Java Programm dar. Beim Aufruf der JVM geben Sie eine Klasse an, in der die JVM nach einer solchen KlassenFunktion sucht – und wenn sie gefunden wird, beginnt dort das Programm. Typischerweise gibt es in einem Java-Programm eine Klasse, die die Anwendung repräsentiert, eine solche „main“ Funktion hat, und den zentralen Einsprungpunkt in die Anwendung darstellt. Achtung – der Quelltext der Klasse MUSS in einer Datei mit dem Namen der Klasse selber und der Extension „java“ stehen. Dies fordert die Sprache! Das gleiche Verfahren gibt es auch noch bei der Verwendung von Packages. Das obige Beispiel muss also in der Datei „Beispiel.java“ stehen. Warum dies so ist, wird in Kapitel 4.5 erklärt. Hinweis – u.a. dieses Beispiel wird im Kapitel 4.3 zur Erklärung der Java-Tools eingesetzt. In Kapitel 4.3.1.1 können sie es im Editor, als Byte-Code und im laufenden Zustand sehen. 3.1.2 Beispiel 2 Um zum einen schon einen Eindruck von der Einfachheit und Leistunsgfähigkeit von Java zu bekommen, und zum anderen zu testen, ob ihre Java Installation problemlos funktioniert hat, hier noch ein zweites einfaches Programm. import javax.swing.JFrame; public class Kap_03_01_Bsp_02_Gui { public static void main(String[] args) { JFrame frame = new JFrame("Mein erstes GUI Fenster"); frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 32 / 409 frame.setLocation(200, 200); frame.setSize(300, 100); frame.setVisible(true); } } Im Gegensatz zum ersten arbeitet es nicht nur auf der Kommandozeile, sondern öffnet zusätzlich ein einfaches leeres GUI-Fenster mit dem Titel „Mein erstes GUI Fenster“ – siehe folgende Abbildung: Abb. 3-1 : einfaches Beispiel GUI-Fenster mit Java Swing Hinweis – u.a. dieses Beispiel wird im Kapitel 4.3 zur Erklärung der Java-Tools eingesetzt. In Kapitel 4.3.1.2 können Sie es im Editor, als Byte-Code und im laufenden Zustand sehen. Hinweis – erzeugte Objekte (hier „frame“) müssen nicht gelöscht werden. Dies macht die JVM automatisch. Der entsprechende Vorgang nennt sich Garbage-Collection, und wird in Kapitel 11.4.3 noch mal kurz angesprochen. Je nachdem, von welcher Programmiersprache Sie kommen, ist dies normal oder ungewöhnlich für Sie. Sprachen wie z.B. C# oder Ruby, haben auch einen Garbage-Collector, und verhalten sich hier wie Java. Kommen Sie dagegen von Sprachen wie z.B. Pascal, C oder C++, so ist dieses Verhalten ungewöhnlich, da in diesen Sprachen der Programmierer Speicher explizit wieder freigeben muss (selbst wenn dies oft nicht direkt zu sehen ist). Java hat hier den Ansatz des Garbage-Collectors gewählt, der zu weniger Fehlern und mehr Komfort für den Programmierer führt. Übrigens: Sie müssen erzeugte Objekte in Java nicht nur „nicht explizit“ löschen, Sie können es natürlich auch gar nicht. Java hat kein Sprachmittel zu Speicherfreigabe – wozu auch? 3.2 Quelltext 3.2.1 Aufbau Java Quelltext ist formlos bzw. Block-, Anweisungs- und Token-orientiert aufgebaut – d.h. nicht zeilen- oder einrückungs-orientiert. Sie können also beliebig Leerzeilen, Leerzeichen, Tabulatoren und Zeilenumbrüche in ihren Quelltext einfügen, solange sie keine Token auseinander reißen. Token sind quasi die kleinste syntaktische Einheit von Java, z.B. Symbole (Namen, Bezeichner,...), Schlüsselwörter (class, import, public,...), Operatoren (+, -, +=, &&, <<<) und Sonderzeichen ({, }, (, ), [,...) © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 33 / 409 3.2.2 Literale: Zahlen, Zeichen und Texte Literale sind Zahlen-, Zeichen- und Text-Konstanten im Quelltext. • Integer Literale sind normale Zahlen. • Gleitkomma-Literale werden an dem Dezimal-Punkt erkannt. • Einzelne Zeichen-Konstanten einfache • Zeichenketten-Konstanten (Texte) werden in doppelte Hochkommata eingeschlossen. Sie müssen spätestens am Zeilenende beendet werden. 2 3.14 .23 2. 'c' "Hallo" "" // // // // // // // Integer Konstante Fliesskomma Konstante Fliesskomma Konstante Fliesskomma Konstante Zeichen-Konstante Zeichenketten-Konstante Leere Zeichenketten-Konstante (Leer-String) 3.2.3 Kommentare Java unterstützt drei Arten von Kommentaren: 1. Bereichs-Kommentare, die mit /* beginnen und mit */ enden, und dabei beliebige Bereiche überdecken dürfen – eine Schachtelung ist nicht erlaubt. 2. Zeilenend-Kommentare, die mit // beginnen und für den Rest der Zeile gelten. 3. Spezielle Java-Doc Kommentare, die mit /** beginnen, mit */ enden und spezielle Tags enthalten – eine Schachtelung ist auch hier nicht erlaubt. Mit dem Tool Java-Doc können aus diesen Kommentaren automatisch Referenz-Dokumente erzeugt werden. Daher werden diese Kommentare häufig auch „Java-Doc Kommentare“ genannt. Java-Doc Kommentare werden in der Vorlesung aus Zeitmangel nicht besprochen. /* Dies ist ein Kommentar */ double d /* Kommentar */ = 3.1; int i = 5; // Noch ein Kommentar Hinweis – die Nutzung von Java-Doc Kommentaren ist sehr zu empfehlen, da auch aktuelle Entwickluns-Umgebungen wie z.B. Eclipse oder NetBeans (siehe Kapitel 4.1.3) diese direkt während der Entwicklung auswerten und z.B. in Views oder Tooltips anzeigen. 3.2.4 Symbole Java unterscheidet bei Symbolen für z.B. Variablen, Funktionen oder Klassen zwischen Großund Kleinschreibung. „Fenster“, „fenster“ und „FENSTER“ sind drei unterschiedliche Symbole. Erlaubte Symbole bestehen u.a. aus den Buchstaben ‚a’ - ‚z’, ‚A’ - ‚Z’, Ziffern und dem Underscore. Hierbei darf das Symbol nicht mit einer Ziffer beginnen. Außerdem sind viele weitere Unicode Zeichen erlaubt, die Buchstaben und Ziffern darstellen. Sie sollten sie in der Praxis (zumindest in unserem Sprachraum) aber nicht benutzen. Aber wenn Sie wollen, können © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 34 / 409 Sie in Java also japanische Variablen- und chinesische Funktions-Namen vergeben – ich kann es aber nicht empfehlen. 3.2.5 Konventionen In Java existiert die Konvention, dass • Package-Namen klein beginnen und klein geschrieben werden, • Klassen- und Interface-Namen groß beginnen und kapitalisiert geschrieben werden, • Funktions- und Variablen-Namen klein beginnen und kapitalisiert geschrieben werden, • Klassen-Konstanten GROSS mit Underscores geschrieben werden. Beispiele: Package-Namen Klassen- bzw. InterfaceNamen Funktions-Namen Variablen-Namen (auch lokale Konstanten) Konstanten-Namen (nur Klassen-Konstanten) mypackage metadataauthoring MyClass MetaDataManager AbstractGuiModel getBackgroundColor calculateAverage pidCount bitLength DEFAULT_COLOR START_WIDTH Selbst wenn sie jetzt noch keine z.B. Packages oder Interfaces kennen, nehmen sie schon mal zur Kenntnis, daß es für alle Namen in Java Konventionen bzgl. der Schreibweise gibt, an die sie sich halten sollten. Damit können wir nun schlussfolgern, dass in der Zeile: System.out.println("Java"); System eine Klasse, out ein Attribut von System, und println eine Elementfunktion von out sein muss. Achtung – in Java sind diese (und andere) Konventionen wichtig, da viele Mechanismen (z.B. bei Beans13) und Tools darauf basieren. Sie sollten sich also konsequent daran halten. Dies betrifft nicht nur die Names-Konventionen, sondern auch andere, die wir noch kennen lernen werden. Viele moderne Entwicklungs-Umgebungen unterstützen Sie durch Warnungen und Hinweise bei der Einhaltung dieser Konventionen – siehe z.B. Kapitel 4.6.3. Java-Beans stellen eine Art wiederverwendbare GUI-Komponenten dar. Aus Zeitmangel werden wir sie in der Vorlesung nicht besprechen können. 13 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 35 / 409 3.3 Ausgabe In einer Applikation kann man Zeichenketten, Konstanten, Variablen, usw. auf die Konsole ausgeben. Dafür existiert in der Klasse „System“ der Ausgabestream (PrintStream) „out“, für den es u.a. die Element-Funktionen „print“ und “println“ gibt. System.out.print(<ein Argument>); System.out.println(<ein Argument>); // Ist ein 'print' inkl. Zeilenumbruch Beide Elementfunktionen erwarten einen Argument beliebigen Typs und geben diesen auf der Konsole aus. Die Elementfunktion „println“ erzeugt zusätzlich zur Ausgabe analog zu „print“ noch einen Zeilenumbruch. int i = 7; System.out.print("->"); System.out.println(i); System.out.println("Java"); System.out.println(42); Ausgabe ->7 Java 42 Den Funktionen können beliebige Argumente übergeben werden. Im folgenden Beispiel sieht man das z.B. an der Ausgabe von „System.out“ – die Ausgabe ist sicher nur bedingt sinnvoll und hilfreich und vor allem ist sie nicht immer gleich (die Zahl hinter dem @ kann anders sein) – aber sie zeigt beispielhaft, dass sich wirklich jedes Argument in einen String wandeln und dann ausgeben läßt. System.out.print(7); System.out.print('c'); System.out.println(78); System.out.println("Hallo"); System.out.println(System.out); // <- es laesst sich wirklich alles ausgeben Mögliche Ausgabe (die genaue Ausgabe von System.out ist nicht definiert) 7c78 Hallo java.io.PrintStream@1a7bf11 Es können auch mehrere Elemente gleichzeitig ausgegeben werden: wenn ein Element ein String ist, wird beim Operator + das andere Element immer automatisch in einen String umgewandelt und danach beide Strings konkateniert (siehe auch Kapitel 9.1.2). int i = 7; System.out.println("Java " + 42 + " " + i); System.out.println(42 + " Java " + i); // Ausgabe: Java 42 7 // Ausgabe: 42 Java 7 Ausgabe Java 42 7 42 Java 7 Bemerkung – die Auswertungsreihenfolge des Operators + ist in Java von links nach rechts definiert. Und auch die Addition zweier Zahlen entspricht der normalen Erwartung. int i = 4; System.out.println(i + 42); System.out.println(4 + 5 + 7); © Detlef Wilkening 1997-2016 // Ausgabe: 46 // Ausgabe: 16 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 36 / 409 Ausgabe 46 16 Der Aufruf von „System.out.println“ ohne Parameter erzeugt einfach nur einen Zeilenumbruch, sprich eine Leerzeile. System.out.println(); // Erzeugt eine Leerzeile 3.4 Strings Neben vielen anderen Typen kennt Java auch einen Datentyp für Texte (Zeichenketten). Dies ist der Typ „String“. String ist kein elementarer Datentyp (vergleiche Kapitel 5.2), kann aber trotzdem einfach benutzt werden. Strings werden im Detail im Kapitel 9.1 besprochen. Wir führen sie hier ganz kurz und pragmatisch ein, da sie der Typ der Kommandozeilen-Argumente (siehe Kapitel 3.5) und der Rückgabe-Typ beim Einlesen einer Zeile von der Kommandozeile (siehe Kapitel 3.10) sind. Strings können mit Literalen initialisiert werden, und können einander zugewiesen werden. Ihre Länge bestimmt man mit der Element-Funktion „length“. String s = "Java"; System.out.println("Laenge von \"" + s + "\" ist " + s.length()); Ausgabe Laenge von "Java" ist 4 3.5 Arrays und Kommandozeilen-Argumente In den Übungs-Aufgaben werden häufiger Kommandozeilen-Argumente benutzt. Sie werden der main-Funktion in einem String-Array übergeben. Um das Array auswerten zu können, benötigen Sie folgende Informationen: • Die Größe des Arrays kann über das Attribut length abgefragt werden – Zugriff über den Punkt-Operator ‚.‘. • Der Zugriff auf die Elemente des Arrays geschieht über den Index-Operator [] (die eckigen Klammern) mit Index in den Klammern. • Arrays sind in Java null-basiert, d.h. z.B. das erste Element hat den Index ‚0‘ Jedes Element des String-Arrays für die Kommandozeilen-Argumente ist ein String (vergleiche Kapitel 3.4), und kann als solcher direkt genutzt werden, oder auch einer String Variablen zugewiesen werden. public class Example { public static void main(String[] args) { if (args.length==0) { System.out.println("Kein Argument"); return; } System.out.println(args.length + " Argument(e)"); System.out.println("- 1. Arg: \"" + args[0] + "\""); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 37 / 409 if (args.length>1) { String arg2 = args[1]; System.out.println("- 2. Arg: \"" + arg2 + "\""); } } } Detaillierter werden Arrays in Kapitel 10 und Strings in Kapitel 9.1 besprochen. Hinweis – der Zugriff auf ein nicht-existentes Array-Element (d.h. Index kleiner Null oder Index zu groß) führt zu einer Exception – siehe Kapitel 3.9 und Kapitel 10.6. Hinweis – u.a. ein Beispiel, das diesem sehr ähnlich ist, wird im Kapitel 4.3 zur Erklärung der Java-Tools eingesetzt. In Kapitel 4.3.4 können Sie es im Editor, als Byte-Code und im laufenden Zustand “sehen“. 3.6 Klassen Klassen sind das elementare Strukturierungsmittel von Java. Pro Datei muss genau eine öffentliche Klasse vorhanden sein, die der Datei ihren Namen gibt. Eine Klasse wird folgendermaßen definiert: [Modifizierer] class <klassen-name> { [Elementfunktionen, Attribute,...] } Eine Klasse wird öffentlich, indem sie den Modifizierer „public“ bekommt. Eine minimale öffentliche Klasse sieht also so aus: public class Klasse { } Die Datei, in der die Klasse definiert ist, muss genauso heißen wie die Klasse - abgesehen von der Dateiextension ‘.java’. Die öffentliche Klasse „Klasse“ muss also in einer Datei „Klasse.java“ stehen. Warum dies so ist, wird in Kapitel 4.5 erklärt. Hinweis – liegt eine Klasse in einem oder mehreren Packages, so muss die Datei in einer Verzeichnisstruktur liegen, die der Package Struktur der Klasse entsprecht. Hinweis – detaillierte Informationen zu Klassen finden sich in den Kapiteln 11 und 12. 3.7 Funktionen Jeder ausführbare Code steht immer in einer Funktion14. Na gut, genau genommen kann Code auch noch in Block-Initialisierern stehen, aber über solch speziellen Sprachmittel wollen wir hier noch nicht reden. 14 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 38 / 409 Funktionen sind immer Bestandteil einer Klasse. Syntax: Modifizierer <Rückgabetyp> <Fkt-Name> ( <Parameterliste> ) { <Implementierung> } Modifizierer – z.B.: public, private, static, final, ... Eine Parameterliste ist eine durch Komma getrennte Auflistung von Parametern (sie kann auch leer sein. Ein Parameter besteht aus Typ und Name – Bsp.: public static void f() { ... } protected final int doit(StringBuffer sb) { ... } private static String calcName(String s, int i) { ... } Aufruf: <Fkt-Name>( <Argumentliste> ); Bsp: f(); calcName("", 1); Rückgaben können beim Funktions-Aufruf ignoriert werden – siehe im Beispiel der Aufruf der Funktion „calcName“, die einen String zurückgibt – der beim Aufruf aber ignoriert wird. Achtung • Die meisten Funktionen in Java sind sogenannte Elementfunktionen. Sie können nur mit Objektbezug aufgerufen werden. • Nur Funktionen mit dem Modifier „static“ (sogenannte Klassenfunktionen) können direkt benutzt werden. • Solange wir noch nicht tiefer in Klassen eingestiegen sind, werden alle unsere selbstgeschriebenen Funktionen Klassen-Funktionen sein – daher den Modifier „static“ enthalten. Bitte denken Sie daran – vergessen Sie das „static“ wird ihr Programm in den meisten Fällen nicht compilieren. Hinweis – detaillierte Informationen zu Funktionen finden sich in den Kapiteln 8 und 12.4. 3.8 Packages Es ist sinnvoll, ihre Programme in Module zu ordnen. Das Sprachmittel hierfür sind Packages. Um eine Klasse in ein Package zu legen, muss die Datei in einem entsprechenden Verzeichnis (mit Namen des Package) liegen, und die Datei muß als erste Anweisung (d.h. nicht Kommentar oder Leerzeilen) eine package-Anweisung enthalten. Ein package-Anweisung beginnt mit dem Schlüsselwort „package“, dann folgt der Package-Name, und das ganze muß mit einem Semikolon abgeschlossen sein. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 39 / 409 package mypackage; Bei verschachtelten Packages müssen auch die Verzeichnisse entsprechend verschachtelt sein. In der Package-Anweisung werden die Package-Namen dann durch Punkte getrennt. package mypackage.nocheins.innerespackage; Analog zu Klassen, deren Namen zu Datei-Namen korrespondieren müssen, müssen also auch die Package-Namen zu Verzeichnissen korrespondieren. Dies verlangt die Sprache, und sie müssen sich daran halten. Warum dies so ist, wird in Kapitel 4.5 erklärt. Hier das „Hallo-Welt“ Beispiel aus Kapitel 3.1.1 in leicht modifizierten Versionen: • Einmal in einem einfachen Package „pack“. • Und dann in einem verschachtelten Package „aussen“ in „mitte“ in „innen“. Beispiel 1 – Achtung, die Datei „Beispiel.java“ muss in einem Verzeichnis „pack“ liegen. package pack; public class Beispiel { public static void main(String[] args) { System.out.println("Hallo Welt"); } } Beispiel 2 – Achtung, die Datei „Beispiel.java“ muss in einem Verzeichnis „innen“ liegen, das in einem Verzeichnis „mitte“ liegen muss, und das wiederum in einem Verzeichnis „aussen“ liegen muss. package aussen.mitte.innen; public class Beispiel { public static void main(String[] args) { System.out.println("Hallo Welt"); } } Hinweis – detaillierter werden Packages in Kapitel 13 besprochen. 3.9 Exceptions Es gibt immer Dinge, die können schief gehen – z.B. eine Eingabe oder ein Array-Zugriff. Solche Probleme werden in Java immer durch Exceptions gemeldet. Da Sie am Anfang sicher viele Fehler machen werden (jeder Fehler ist gut, denn er zeigt, dass Sie Java angewendet haben), werden Sie in Java von Anfang an häufig mit Exceptions konfronitert werden. Im Augenblick reicht uns – neben dem Thema „Checked-Exceptions“ – siehe gleich – hier das Wissen, dass mit Exceptions Fehler in unserem Programm angezeigt werden. Irgendetwas haben Sie falsch gemacht, oder ist einfach schief gelaufen. Am Anfang gehen wir erstmal davon aus, dass alles gut geht – außer wir wollen den Fehlerfall explizit nutzen, wie z.B. in © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 40 / 409 Kapitel 3.11. Daher könnten wir Exceptions eigentlich erstmal ignorieren, aber... Aber ein Teil der Exceptions können in Java nicht ignoriert werden, nämlich alle CheckedExceptions. Daher: immer wenn eine Funktion ein „Problem“ mit einer solchen CheckedException melden könnte, dann müssen sie sich darum kümmern. Ein Beispiel dafür ist die Funktion „read“ in der Anweisung „System.in.read()“ im nächsten Kapitel. Schreiben Sie im Augenblick die problematische(n) Anweisung(en) einfach in einen try-Block und fügen noch einen leeren catch-Block an. Außerdem merken sie sich, dass der Programmfluss im Fehlerfall in den catch-Block verzweigt. Beispiele hierfür finden wir gleich in den Kapiteln 3.10 und 3.11. System.out.println("vor try"); try { System.out.println("vor read"); // "read()" kann schief gehen – d.h. koennte eine Exception werfen System.in.read(); System.out.println("nach read"); } catch (Exception x) { // Hierhin verzweigt der Programmfluss im Fehlerfall System.out.println("Fehlerbehandlung"); } System.out.println("nach try/catch"); Ein zweiter Fall, in dem wir uns auch im Augenblick schon um Exceptions kümmern müssen, ist wenn Fehler möglich sind, und diese durch Exceptions gemeldet werden. Dies findet sich z.B. bei Konvertierungen (siehe Kapitel 3.11) oder bei fehlerhaften Array-Zugriffen (siehe Kapitel 3.5 und Kapitel 4.13). Hinweis – detaillierter werden Exceptions in Kapitel 22 besprochen. 3.10 Eingabe Die Eingabe von der Kommandozeile war unter Java lange Zeit recht kompliziert, da mehrere Streams und Reader miteinander verbunden werden mußten, und Checked-Exceptions beteiligt waren. Prinzipiell ist der Mechanismus mit den Streams und Readern sehr leistungsfähig – nur für einen Anfänger und eine so alltägliche Aufgabe nicht angemessen – wir werden Streams und Reader in Kapitel 23 kennen lernen. Mit dem JDK 1.5 konnte das Einlesen mit Hilfe der Klasse „Scanner“ und der Element-Funktion „nextLine()“ leicht vereinfacht werden. Seit dem JDK 1.6 kann direkt auf die Kommandozeile zugegriffen und eine Zeile als String eingelesen werden. Das folgende Beispiel (für das JDK 1.6) zeigt, wie man eine Zeile als String von der Kommandozeile einliest und auswertet. Ignorieren Sie erstmal die Dinge, die wir noch nicht eingeführt haben wie z.B. die Kontrollstrukturen „while“, „if“ und „break“. Der grundsätzliche Ablauf des Programms sollte auch so klar sein. Es ist ein Echo-Programm, das alle Eingaben des Nutzers direkt wieder in doppelten Anführungs-Zeichen auf der Kommandozeile ausgibt – mit der Information wieviele Zeichen eingelesen wurden. public class Chapter0310Ex01 { © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 41 / 409 public static void main(String[] args) { System.out.println("Echo-Programm - JDK 1.6"); while (true) { System.out.print("> "); String in = System.console().readLine(); if (in.length() == 0) { break; } System.out.println(" \"" + in + "\" - " + in.length() + " Zeichen"); } System.out.println("Programm-Ende"); } } mögliche Ausgabe Echo-Programm - JDK 1.6 > Hallo "Hallo" - 5 Zeichen > Ich lerne jetzt Java "Ich lerne jetzt Java" - 20 Zeichen > Programm-Ende Nehmen Sie das Beispiel erstmal so hin, falls nicht alles klar ist – im Laufe der Vorlesung werden sich die einzelnen Fragen dazu aufklären. Und wenn sie Eingabe machen müssen, benutzen sie das Beispiel ganz pragmatisch als Vorlage und passen es im Rahmen ihrer Kenntnisse und Bedürfnisse an. Die jeweils eingelesene Zeile findet sich innerhalb der WhileSchleife im String „in“, und kann dann von ihnen genutzt werden. Achtung – das obige Beispiel funktioniert problemlos auf der Konsole, aber nicht in der Eclipse. Die Eclipse startet ihr Java-Programm im Hintergrund mit einer Umgebung ohne echter Konsole. Der Aufruf von „console()“ liefert dort „null“ (siehe Kapitel todo) zurück und erzeugt dann eine Null-Pointer-Exception. Dies ist ein bekannter Bug in der Eclipse, der auch in Version 4.5.2 (Mars Update 2) noch existiert. • https://bugs.eclipse.org/bugs/show_bug.cgi?id=122429 • http://stackoverflow.com/questions/104254/java-io-console-support-in-eclipse-ide Darum sind alle Beispiele in diesem Tutorial noch mit dem folgenden JDK 1.1 Mechanismus umgesetzt. Aus historischen Gründen – und da Ihnen dieser Code in der Praxis noch häufig begegnen könnte (z.B. hier im Tutorial, da ich die Eclipse benutze) – hier auch die Beispiele aus ganz alten Zeiten (ab JDK 1.1) und für das JDK 1.5. Die Programme machen genau das Gleiche wie das erste Beispiel – sind nur viel mehr Code und schwerer zu verstehen. Das Beispiel für das JDK 1.1 zeichnet sich dadurch aus, dass • es Streams & Reader benutzt (siehe Kapitel 23), und • eine Checked-Exception abfangen muss, die von „reader.readLine()” geworfen werden könnte. import java.io.InputStreamReader; import java.io.BufferedReader; public class Chapter0310Ex02 { © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 42 / 409 public static void main(String[] args) { try { System.out.println("Echo-Programm - JDK 1.1"); InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); while (true) { System.out.print("> "); String in = reader.readLine(); if (in.length()==0) { break; } System.out.println(" \"" + in + "\" - " + in.length() + " Zeichen"); } } catch (Exception x) { System.out.println("Unerwarteter Fehler"); } System.out.println("Programm-Ende"); } } Hinweis – da Einlesen schief gehen kann sind hier Exceptions möglich, und hier sind diese Checked-Exceptions und müssen daher abgefangen werden – vergleiche Kapitel 3.9. Sie können ja mal den try-catch-Block weg lassen – dann wird der Compiler einen Compiler-Fehler melden. Die dritte Implementierung unseres Echo-Programms basiert auf dem JDK 1.5 und nutzt einen „Scanner“. Da Scanner geschlossen werden müssen und unser Beispiel 100% korrekt sein soll, müssen wir auch die Fälle berücksichtigen, wo in unserem Programm etwas schief geht und dieses Problem durch eine Exception gemeldet wird (vergleiche Kapitel 3.9). Dadurch gewinnen wir wieder etwas Exception-Handling, diesmal in Form eines Try/Finally-Blocks: import java.util.Scanner; public class Chapter0310Ex03 { public static void main(String[] args) { System.out.println("Echo-Programm - JDK 1.5"); Scanner sc = new Scanner(System.in); try { while (true) { System.out.print("> "); String in = sc.nextLine(); if (in.length() == 0) { break; } System.out.println(" \"" + in + "\" - " + in.length() + " Zeichen"); } } finally { sc.close(); } System.out.println("Programm-Ende"); } } Sie sehen, nutzen Sie moderne JDKs und bleiben beim Code des ersten Beispiels für das Einlesen von der Kommandozeile. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 43 / 409 3.11 Konvertierungen In den Übungs-Aufgaben werden ab und zu Konvertierungen von Strings zu Ganz- oder Gleitkomma-Zahlen benötigt. Diese Konvertierungen können über Klassen-Funktionen der Klassen „Integer“ und „Double“ vorgenommen werden. Achtung – die Konvertierungen können natürlich schief gehen – in diesem Fall wirft die Parse-Funktion eine „NumberFormatException“ Exception. Auch dies ist also ein Beispiel dafür, dass Sie sich in Java immer wieder um Exceptions kümmern müssen – auch wenn wir sie noch gar nicht richtig kennen. public class Example { public static void main(String[] args) { if (args.length==0) { System.out.println("Kein Argument"); return; } System.out.println("Arg: " + args[0]); try { int i = Integer.parseInt(args[0]); System.out.println("i: " + i); } catch (NumberFormatException x) { System.out.println(args[0] + " laesst sich nicht in int wandeln"); } try { double d = Double.parseDouble(args[0]); System.out.println("d: " + d); } catch (NumberFormatException x) { System.out.println(args[0] + " laesst sich nicht in double wandeln"); } } } Hinweis – die Klassen „Integer“ und „Double“ sind quasi die Objekt-Analogien zu den elementaren Typen „int“ und „double“. Sie werden z.B. als Wrapper-Klassen für Container benutzt, und sie bieten allgemeine Hilfs-Funktionen wie z.B. „parseInt“ für den jeweiligen Typ an. Näheres zu diesen Wrapper-Klassen finden Sie in Kapitel 9.4. 4 Praktikum 4.1 Tools Für die Entwicklung von Java-Applikationen bzw. Applets werden zwingend ein Editor, ein Java Compiler und eine virtuelle Java-Maschine (JVM – Java Virtual Machine) benötigt. Es gibt viele weitere Tools, die das Entwickler-Leben erleichtern können, die aber nicht zwingend notwendig sind – z.B. Debugger, Test-Werkzeuge, Versions-Verwaltungen, Differ, und und und. Heutzutage wird ein Java Programm aber typischerweise in einer IDE (Integrierte Entwicklungs Umgebung) wie Eclipse, NetBeans oder IntelliJ entwickelt. Auch wir werden im Praktikum mit einer IDE (konkret „Eclipse“) arbeiten – siehe Kapitel 4.1.3 und 4.6. Trotzdem sind auch die Kommandozeilen-Tools in der Praxis sehr wichtig, und sie fördern ein Verständnis für die Mechanismen, die in einer IDE unter der Haube ablaufen. Daher wollen wir uns vor den IDEs © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 44 / 409 kurz die JDK Entwicklungs-Tools anschauen – siehe Kapitel 4.1.1 und 4.3. 4.1.1 Das JDK und die Tools „java“ und „javac“ Das JDK (siehe auch Kapitel 2.3) von Oracle enthält neben der eigentlichen Java VM auch viele Tools zur Programmierung mit Java. Da viele diese Tools reine KommandozeilenWerkzeuge sind, ist ihre Benutzung – gerade für den Anfänger – nicht besonders einfach und komfortabel. Dafür sind sie kostenlos u.a. im Internet erhältlich, und können sowohl privat als auch komerziell genutzt werden. Die einfachste Lösung zur Entwicklung von Java-Programmen wäre also ein beliebiger Editor mit den Kommandozeilen-Werkzeugen „javac“ und „java“ von Oracle. • Der Java-Compiler „javac“. Mit diesem Programm wird aus dem oder den Java-Quelltexten der Java Byte-Code erzeugt. Java Byte-Code hat immer die Extension „.class“. • Die Java virtuelle Maschine („JVM“) „java“. Mit diesem Programm wird der Java Byte-Code ausgeführt. Sie benötigen z.B. das aktuelle Java SDK: J8SE SDK (Java 8 Standard Edition – Software Development Kit) – die aktuelle Version ist 1.8.0_u77, d.h. JDK 1.8 Update 77 (Stand Anfang April 2016). Es empfiehlt sich auf jeden Fall auch die „J2SE 8.0 Documentation“ mit herunterzuladen. Achtung – die Screenshots beziehen sich noch auf Version 1.8.0_u40. Homepage bzw. Download-Adresse • http://www.oracle.com/de/index.html • http://www.oracle.com/de/technologies/java/index.html • http://www.oracle.com/technetwork/java/javase/downloads/index.html © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 45 / 409 Abb. 4-1 : Oracle Homepage – Einstieg in den Download des JDK © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 46 / 409 Abb. 4-2 : Oracle Java Download Seite – JDK Auswahl © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 47 / 409 Abb. 4-3 : Oracle JDK Download Seite – Plattform auswählen und downloaden Zusätzlich zum aktuellen JDK sollten Sie immer auch die aktuelle Java-Dokumentation auf der Oracle Java Download Seite (weiter unten) herunterladen. Und wenn Sie mit JavaFX arbeiten – siehe Kapitel 16 – dann auch die JavaFX Dokumentation. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 48 / 409 Abb. 4-4 : Oracle Java Download Seite – JDK Dokumentation Auswahl Abb. 4-5 : Oracle JDK Doku Download Seite – Doku auswählen und downloaden Hinweis – auf der Oracle Download Seite können Sie übrigens auch in einem immer die jeweils aktuelle NetBeans IDE mit herunterladen – siehe auch Kapitel 4.1.3.2. Nach der Installation des JDK steht Ihnen die JVM direkt z.B. auf der Kommandozeile zur Verfügung und kann mit „java“ aufgerufen werden. Mit der Option „-version“ z.B. gibt die JVM © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 49 / 409 nur ihre Version aus und beendet sich dann direkt. Die Ausführung von echten Java Programmen auf der Kommandozeile lernen wir in Kapitel 4.3. Abb. 4-6 : Installiertes JDK – Aufruf der JVM Wollen Sie auch den Java-Compiler „javac“ und die anderen JDK-Tools direkt nutzen können, so müssen Sie z.B. unter Windows in der System-Steuerung den Pfad zu dem Bin-Verzeichnis Ihrer JDK-Installation der Umgebungs-Variablen „path“ hinzufügen. Bei einer „normalen“ Installation ist dies unter einer 64 Bit Windows Version: „C:\Program Files\Java\jdk1.8.0_77\bin“. Wenn Sie dies gemacht haben, dann können Sie auch den JavaCompiler „javac“ direkt auf der Kommandozeile aufrufen – z.B. auch mit der Option „-version“. Abb. 4-7 : Aufruf des Java-Compilers nach gesetzter Path Umgebungs-Variable Haben Sie das JDK Bin-Verzeichnis nicht in den Pfad aufgenommen, dann müssen Sie beim Aufruf den kompletten Pfad-Namen inkl. Extension mitangeben: © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 50 / 409 Abb. 4-8 : Aufruf des Java-Compilers mit vollem Pfad Werden Sie hauptsächlich oder vielleicht ausschließlich mit einer IDE arbeiten, so lohnt sich das Setzen der Umgebungs-Variablen nicht. Arbeiten Sie häufig auf der Kommando-Zeile, so rentiert sich das Setzen sehr schnell. Hinweis – die Nutzung der JDK Tools ist in Kapitel 4.3 beschrieben. Die hoffentlich mit heruntergeladene Java-Dokumentation müssen Sie nur entpacken. Darin finden Sie mit „index.html“ den Einstiegs-Punkt in die sehr umfangreiche Dokumentation: • Die Java Dokumentation „jdk-8u77-docs-all.zip“ entpackt sich nach „docs“ • Die JavaFX Dokumentation „javafx-8u77-apidocs.zip entpackt sich nach „api“ Abb. 4-9 : Entpackte Java Dokumentation © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 51 / 409 Abb. 4-10 : Java Dokumentations Einstieg „index.html“ 4.1.2 Editor ConTEXT oder Notepad Für das Arbeiten ohne IDE – was in der Praxis immer wieder sinnvoll ist – empfiehlt sich ein guter Editor. Während die typische Linux Distribution meist einige sehr mächtige Editoren enthält, bringt Windows von sich aus nur den Editor „Notepad“ mit, der leider nicht empfehlenswert ist. Zum Glück gibt es für Windows viele freie Editoren – ganz gut leben kann man z.B. mit: • ConTEXT http://www.contexteditor.org/index.php • Notepad++ http://notepad-plus-plus.org/ • jedit http://www.jedit.org/ Aber zweifelsfrei gibt es viele weitere empfehlenswerte Editoren unter Windows. 4.1.3 IDEs Natürlich gibt es für die Java Entwicklung auch integrierte grafische Entwicklungs-Umgebungen (IDEs15), die alle notwendigen Werkzeuge (Editor, Compiler, JVM) und viele weitere Tools und Hilfen unter einem Dach zusammenfassen. Die Benutzung dieser Umgebungen ermöglicht ein weitaus schnelleres und komfortableres Arbeiten, da viele Arbeiten automatisiert sind bzw. in einer grafischen Umgebung erfolgen können. Ein Teil dieser IDEs sind als „Personal Version“ für den privaten Gebrauch kostenlos erhältlich. Andere sind komplett frei, d.h. auch für den komerziellen Gebrauch – darunter fallen z.B. die IDEs Eclipse von IBM oder Net-Beans von Oracle, die beide sehr empfehlenswert sind. 15 Integrated Development Environment © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 52 / 409 4.1.3.1 Eclipse Eclipse etabliert sich im Augenblick immer mehr als die zentrale Open-Source Umgebung für die Java Entwicklung. Dies liegt u.a. daran, dass allein IBM über 160 Entwickler16 für die (Weiter-) Entwicklung von Eclipse einsetzt, und sie trotzdem kostenlos zur Verfügung stellt. Außerdem gibt es eine große Eclipse Community17, die viele Erweiterungen (Plugins) und Hilfen bereit stellt. Unterm Strich ist Eclipse mittlerweile ein kleines Monster, eine IDE mit einer unglaublichen Vielzahl an Möglichkeiten und Features. Wir werden Eclipse im Praktikum benutzen – die aktuelle Version ist 4.5.2 Mars (Stand Anfang April 2016). Achtung – die Screenshots beziehen sich noch auf die ältere Eclipse Version 4.4.2 Luna. Homepage bzw. Download-Adresse • http://www.eclipse.org • http://www.eclipse.org/downloads/ Achtung – die Eclipse IDE gibt es in mehreren Varianten. Für den Anwendungsfall der Vorlesung, d.h. für die Entwicklung von Java Applets und Java Applikationen für den Desktop ist die „Eclipse IDE for Java Developers“ die richtige Wahl. Abb. 4-11 : Eclipse Homepage – Wechsel zur Download Seite Darunter so berühmte Leute wie Erich Gamma, den Autor des Buchs „Entwurfsmuster“ – siehe Kapitel todo. 17 Dies äußert sich in Deutschland z.B. dadurch, dass es eine eigene Eclipse Zeitschrift gibt, oder auch Konferenzen zum Thema Eclipse. 16 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 53 / 409 Abb. 4-12 : Eclipse Download-Seite – Eclipse IDE for Java Developers Nach der Installation von Eclipse müssen Sie die heruntergeladene Zip-Datei nur auspacken. In dem Ziel-Verzeichnis finden Sie dann die Datei „eclipse.exe“, mit der Sie die Eclipse starten können. Die Nutzung der Eclipse ist in Kapitel 4.6 beschrieben. Hinweis – Eclipse selber ist in Java geschrieben. Sie benötigen also ein installiertes JRE auf Ihrem Rechner. Da Sie ja mit Java entwickeln wollen, sollten Sie also das aktuelle JDK installiert haben – und das enthält auch das jeweilige JRE. 4.1.3.2 NetBeans Natürlich hat auch Eclipse seine Schwächen, und andere IDEs und Tools ihre Stärken. Die Vorlesung möchte keine Werbung für Eclipse sein. Schauen Sie sich ruhig die Alternativen gut an, und entscheiden Sie nach Ihren eigenen Vorstellungen und Wünschen. Eine vergleichbare Alternative mit manchen Vor- aber auch Nachteilen ist sicher NetBeans von Oracle – URL siehe Kapitel 4.1.1. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 54 / 409 Hinweis – NetBeans selber ist in Java geschrieben. Sie benötigen also ein installiertes JRE auf Ihrem Rechner. Da Sie ja mit Java entwickeln wollen, sollten Sie also das aktuelle JDK installiert haben – und das enthält auch das jeweilige JRE. 4.2 Source Organisation Bevor wir unsere ersten Programme übersetzen und laufen lassen, noch einmal die folgenden sehr wichtigen Hinweise: • Die gesamte programmierte Logik in einem Java Programm spielt sich in Klassen ab. • Jeder Java Quelltext muss genau eine public Klasse (oder Interface) enthalten. • Der Name der Klasse muss dem Namen der Quelldatei entsprechen (inkl. Groß- und Kleinschreibung). Die Datei muss zusätzlich die Extension „.java“ haben. - Beispiel: Die public Klasse „Pop3Protocol“ muss zwingend in einer Datei „Pop3Protocol.java“ stehen. • Liegt die Klasse in einem Package, so muss die Datei in einem Verzeichnis mit dem entsprechenden Package Namen liegen (auch hier gilt natürlich wieder Gross- und Kleinschreibung). - Beispiel: Liegt die public Klasse „TestCase“ in einem Package „applicationlogic“, so muss die Datei „TestCase.java“ zwingend in einem Verzeichnis „applicationlogic“ liegen. - Beispiel: Liegt die public Klasse „SeqNode“ in den Packages „db“ und „implementation“ (d.h. das Package „db“ enthält das Package „implementation“), so muss die Datei „SeqNode.java“ in einem Verzeichnis „implementation“ liegen, und dieses wiederum in einem Verzeichnis „db“. • Der Grund für die Korrelation von Klassen- und Datei-Namen, bzw. von Package- und Verzeichnis-Namen wird in Kapitel 4.5 erklärt. Die Korrelation von Klassen- zu Datei-Namen, und Package- zu Verzeichnis-Namen wurde zwar schon in den Kapiteln 3.1 und 3.8 erwähnt, aber die Erfahrung zeigt, dass dies ein ganz typischer Anfänger-Fehler ist. Darum sollte es hier noch mal erwähnt sein. Passen Sie also darauf auf. In den letzten Jahren waren sicher die Hälfte der Probleme in den ersten Praktikumswochen auf diesen Fehler zurückzuführen. 4.3 Benutzung der JDK Entwicklungs-Tools Für die Beispiele hier wird davon ausgegangen, dass unter Windows gearbeitet wird, das aktuelle JDK 1.8 installiert ist, und für z.B. den Java-Compiler das JDK Bin-Verzeichnis in die Path Umgebungs-Variable eingetragen wurde – siehe Kapitel 4.1.1. 4.3.1 Java mit Klassen ohne Packages Liegen die Java Klassen in keinem Package, so ist die Benutzung der Kommandozeilen-Tools sehr einfach. Schauen wir uns die die Schritte zur Verarbeitung und Ausführung des „Hallo- © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 55 / 409 Welt“ Beispiels aus Kapitel 3.1.1 an. Achtung – Klassen, die in keinem Package liegen, sollten die absolute Ausnahme sein. Sie sind eine schnelle Lösung für kleine Test- oder Beispiel-Programme – von daher werden Sie solche viel in diesem Tutorial finden. Für mehr sollte man sie aber nicht nutzen. Später – wenn Sie reale Programme schreiben – sollten Sie für alle ihre Programme eine sinnvolle DateiOrganisation anlegen, die sich dann in den Packages und Verzeichnissen widerspiegelt. 4.3.1.1 „Hallo Welt“ Beispiel Erstellen Sie mit einem Editor den Quelltext des „Hallo-Welt“ Java Programms aus Kapitel 3.1.1 – diesmal aber mit dem Klassen-Namen „HalloWelt“, und speichern Sie ihn unter dem Namen „HalloWelt.java“ ab. Bei mir liegt die Datei z.B. direkt unter „E:\java“. Abb. 4-13 : „Hallo Welt“ Quelltext im Editor (Datei „e:\java\HalloWelt.java“) Achtung – passen Sie auf, dass die Klasse genauso wie die Datei heißt (bis auf die zusätzliche Extension „.java“) – die Sprache Java fordert dies, siehe Kapitel 3.6. Öffnen Sie jetzt eine Kommandozeile für dieses Verzeichnis. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 56 / 409 Abb. 4-14 : Kommandozeile für das Verzeichnis „e:\java“ Rufen Sie den Java Compiler „javac“ mit Angabe der zu übersetzenden Datei auf. In Abhängigkeit von ihrer Konfiguration müssen Sie entweder den Pfad zum Compiler mit angeben, oder können ihn weglassen – siehe Kapitel 4.1.1. Bei mir ist der Pfad zu den JDK Entwicklungs-Tools in der Pfad-Umgebungsvariable enthalten – ich brauche den Pfad also nicht anzugeben: > javac HalloWelt.java Der Java Compiler compiliert die angegebene Datei und erzeugt den entsprechenden ByteCode (Datei „HalloWelt.class“) im aktuellen Verzeichnis. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 57 / 409 Abb. 4-15 : Der Java Compiler erzeugt den Byte-Code des „Hallo Welt“ Beispiels Diesen müssen wir jetzt in der virtuellen Java Maschine (JVM) ausführen. Dazu rufen wir die virtuelle Maschine „java“ auf, und übergeben den vollständigen Namen der Klasse18. > java HalloWelt Abb. 4-16 : Die virtuelle Java Maschine (JVM) führt das „Hallo Welt“ Beispiel aus Nicht den Namen der Class-Datei oder den Namen der Quelltext-Datei, sondern wirklich den vollständigen Namen der Klasse. Dies impliziert, dass bei Klassen die in Packages liegen (siehe Kapitel 4.3.2.2) natürlich auch alle Packages mit angegeben werden müssen. 18 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 58 / 409 4.3.1.2 GUI Beispiel Vergleichbar zum „Hallo-Welt“ Beispiel läuft Handling des GUI Beispiels aus Kapitel 3.1.2 ab. Abb. 4-17 : GUI Quelltext im Editor (Datei „e:\java\Gui.java“) Abb. 4-18 : Der Java Compiler erzeugt den Byte-Code des GUI Beispiels © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 59 / 409 Abb. 4-19 : Die virtuelle Java Maschine (JVM) führt das GUI Beispiel aus 4.3.1.3 Mehrere Klassen gleichzeitig compilieren Wollen Sie mehrere Dateien auf einmal übersetzen, so können Sie den Wildcard „*“ im DateiNamen benutzen. Löschen Sie z.B. die beiden gerade erzeugten Class-Dateien, und compilieren Sie unsere beiden Beispiel-Quelltexte „HalloWelt.java“ und „Gui.java“ in einem Rutsch neu: > javac *.java Abb. 4-20 : Der Java Compiler erzeugt den Byte-Code aller Klassen im Verzeichnis © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 60 / 409 4.3.2 Java mit Klassen in Packages Liegen die Klassen in Packages, so wird das ganze etwas aufwändiger, aber nicht wirklich kompliziert. Um das zu lernen, schreiben wir ein Beispiel ähnlich dem „Hallo-Welt“ von eben, nur dass die Klasse jetzt in den Packages „aussen.mitte.innen“ liegt. Natürlich legen wir die Datei korrekt in die Verzeichnis-Struktur „aussen\mitte\innen“19. Abb. 4-21 : Package Quelltext im Editor (Datei „PackageBeispiel.java“) Damit das Beispiel auch später noch nutzbar ist, wenn wir die Source- und Class-Path Optionen erklären (siehe Kapitel 4.3.2.3 und Kapitel 4.5), legen wir die Verzeichnis-Struktur „aussen\mitte\innen“ nicht direkt in „e:\java“ an, sondern „betten“ sie noch zusätzlich in ein Zwischen-Verzeichnis „Sourcen“ ein. So sieht unsere Struktur dann aus: Abb. 4-22 : Verzeichnis-Struktur für das Package Beispiel Erinnern sie sich? Packages müssen mit den Verzeichnissen korrespondieren – siehe Kapitel 3.8 und todo. 19 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 61 / 409 4.3.2.1 Einfache, aber falsche Vorgehensweise Wenn Sie jetzt einfach die Vorgehensweise von eben übertragen, erleben Sie eine kleine Überraschung. Aber warum nicht – probieren wir es einfach mal aus: • Wir wechseln mit der Konsole in das Verzeichnis „innen“ • Wir rufen den Java Compiler auf • Und wir starten das Programm... Abb. 4-23 : Der Java Compiler erzeugt den Byte-Code des Package Beispiels Wie wir sehen, compilierte der Compiler den Quelltext ganz problemlos, und der Byte-Code liegt damit vor. Also starten wir das Programm mal: Abb. 4-24 : Fehler bei der Ausführung des Package Beispiels Was ist das? Unser Programm startet nicht – statt dessen meldet uns die JVM einen Fehler. Um das Problem vollständig zu verstehen, fehlt uns noch etwas Wissen. Im Augenblick soll die © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 62 / 409 Bemerkung reichen, dass wir versucht haben, den Compiler und vor allem die JVM zu betrügen. Beim Starten des Programms erzählen wir der JVM von einer Klasse „Beispiel“ ohne Packages, die aber in Packages liegt. Belassen wir es dabei – keine Details, die kommen später – lernen wir lieber, wie man es richtig macht. Aber vorher löschen Sie den fehlerhaft erzeugten Byte-Code „PackageBeispiel.class“, bevor er uns noch mehr Ärger macht. 4.3.2.2 So ist es richtig Statt in das Verzeichnis mit dem Java Quelltext zu wechseln, müssen Sie bei der Übersetzung das Wurzel-Verzeichnis für die Quelltexte angeben. Das Wurzel-Verzeichnis ist quasi das Verzeichnis, das das äußere Package enthält, bei uns also „e:\java\Sourcen“. Wechsel wir also zurück in unser Java-Verzeichnis. Wären wir dort also einfach mit unserer Konsole geblieben, hätten wir uns viel vergebene Arbeit sparen können. Zurück in „e:\java“ werfen wir den Compiler an, und geben jetzt beim Compiler-Aufruf zwei weitere Optionen an: • Die Option „-sourcepath“ gibt das (oder die) Wurzel-Verzeichnis Ihrer zu übersetzenden Klassen an – in unserem Beispiel „Sourcen“ als relative Pfad-Angabe oder „e:\java\Sourcen“ als absolute Pfad-Angabe. • Die zu übersetzende Java-Datei (oder die Java-Dateien) müssen wir jetzt natürlich inkl. Pfad angeben. > javac -sourcepath Sourcen Sourcen\aussen\mitte\innen\PackageBeispiel.java Abb. 4-25 : Der Java Compiler erzeugt den Byte-Code mit „sourcepath“ Option Hinweise: © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 63 / 409 • Natürlich kann man auch hierbei Wildcards benutzen, um alle Dateien gleichzeitig zu compilieren. • Stehen Sie während des Compilierens im Wurzel-Verzeichnis Ihres Programms, so geben Sie als Soure-Path einfach nur den Punkt „.“ an, der ja für das aktuelle Verzeichnis steht. Für die Ausführung wechseln wir nun in das Wurzel-Verzeichnis unseres Programms, d.h. in „e:\java\Sourcen“. Wir werden gleich noch lernen, wie man das Programm aus einem beliebigen Verzeichnis ausführt – aber erstmal einfach: Beim Aufruf der JVM muss der vollständige Klassen-Name der Main-Klasse mitgegeben werden. Das war auch eben schon so, aber eben lag die Klasse nicht in einem Package – von daher war der einfache Klassen-Name auch gleichzeitig der vollständige Klassen-Name. Jetzt, mit Packages, ist der vollständige Klassen-Name der Klassen-Name inkl. Package-Namen – getrennt durch Punkte, d.h.: „aussen.mitte.innen.PackageBeispiel“. > cd Sourcen > java aussen.mitte.innen.PackageBeispiel Abb. 4-26 : Korrekte Ausführung des Package Beispiels 4.3.2.3 Und was, wenn man nicht im Wurzel-Verzeichnis steht? Lassen Sie uns wieder zurück in unser Java-Verzeichnis „e:\java“ wechseln. Nun stehen wir nicht mehr im Wurzel-Verzeichnis unseres Programms – wie starten wir es denn nun? Wenn wir hier die JVM mit der Main-Klasse aufrufen bekommen wir nur einen Fehler: Abb. 4-27 : Fehler bei Ausführung außerhalb des Wurzel-Verzeichnisses © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 64 / 409 Das war auch zu erwarten. Woher soll die JVM wissen, wo unser Programm auf der Platte liegt. Wir stehen quasi in einem beliebigen Verzeichnis – und in einem beliebigen anderen Verzeichnis liegt unser Programm. In diesem Fall, muß man – analog zur Option „-sourcepath“ beimm Compiler – nun der JVM das Wurzel-Verzeichnis angeben. Nur heißt die Option hier „-classpath“ oder kurz „-cp“, da sich die JVM nicht mehr für die Sourcen interessiert, sondern statt dessen den Klassen-Pfad für den Byte-Code kennen muss. > java -cp Sourcen aussen.mitte.innen.PackageBeispiel Abb. 4-28 : Korrekte Ausführung des Package Beispiels mit Class-Path Option „-cp“ Achtung – genauso wie die Sourcen definiert heißen und an genau definierten Stellen (relativ zum Source-Wurzel-Verzeichnis) liegen müssen, so muß auch der Byte-Code genau definiert heißen und relativ zu einer Klassen-Pfad-Wurzel genau definiert liegen. Also benennen Sie niemals Byte-Code um bzw. verschieben Sie ihn auch nicht, bzw. nur sehr bewußt. 4.3.3 Trennung von Sourcen und Byte-Code Nun ist die Vermischung von Source-Code und Byte-Code im gleichen Verzeichnis keine gute Idee – oder sagen wir allgemein: die Vermischung von User-erzeugten und automatisch generierten Dateien. Schön wäre es, wenn der Java-Compiler den Byte-Code in ein extra Verzeichnis legen würde. Dies kann mit der Option „-directory“ oder kurz „-d“ erreicht werden. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 65 / 409 Abb. 4-29 : Ausgangs Struktur für Byte-Code eigene Verzeichnisse Achtung, - das Ausgabe Verzeichnis (hier „output“) muss existieren. Beispielhaftes Compilieren aus dem Verzeichnis „e:\java“ heraus. > javac -sourcepath sourcen -d output sourcen\aussen\mitte\innen\*.java Abb. 4-30 : Der Java Compiler erzeugt den Byte-Code mit Trennung des Byte-Codes Der Compiler erzeugt nun beim Compilieren nicht nur den Byte-Code, sondern auch die den Packages bzw. dem Source-Path entsprechende Verzeichnis-Struktur. D.h. im AusgabeVerzeichnis liegt nicht einfach die Byte-Code Datei „PackageBeispiel.class“. Statt dessen hat der Compiler auch die Verzeichnisse „aussen“, „mitte“ und „innen“ angelegt, und erst im Verzeichnis „innen“ liegt der Byte-Code. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 66 / 409 Abb. 4-31 : Erzeugte Verzeichnis Struktur und Byte-Code Denken Sie jetzt beim Starten des Programms nur daran, dass Sie den Class-Path (hier z.B. „output“), und nicht den Source-Path (hier z.B. „sourcen“) angeben müssen. Die Sourcen und der Byte-Code liegen jetzt ja in zwei unterschiedlichen Verzeichnis-Strukturen. > java -cp output aussen.mitte.innen.PackageBeispiel Abb. 4-32 : Korrekte Ausführung des Package Beispiels mit Trennung des Byte-Codes Hinweis – wenn Sie so arbeiten, d.h. mit der Trennung des Source-Codes vom Byte-Code, dann müssen Sie immer nur Ihr Source-Verzeichnis sichern, und haben damit immer alles im Griff. Der Byte-Code läßt sich ja jederzeit wieder herstellen. 4.3.4 Kommandozeilen-Argumente Sie können an ein Java Programm Argumente übergeben, indem Sie diese als letzte Argumente – d.h. nach dem Klassen-Namen – an die JVM übergeben. Unser Beispiel ist der Einfachheit halber ohne Packages, und lehnt sich an das Beispiel aus Kapitel 3.5 an. Achtung, das Beispiel soll mit zwei Kommandozeilen-Argument aufgerufen werden: © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 67 / 409 Abb. 4-33 : Kommandozeilen Argument Quelltext im Editor (Datei „e:\java\Args.java“) > javac Args.java > java Args > java Args Hallo > java Args Hallo Java Abb. 4-34 : Compilation und Ausführung des Kommandozeilen Argumente Beispiels 4.4 Projekt Tools Nehmen wir mal an, Sie entwickeln ein großes Projekt, mit vielen Packages und noch mehr Klassen. Das neben Quelltexten auch noch viele andere Dinge wie z.B. Icons, Bilder, Videos und lokalisierte Textdateien enthält. Und wo es nicht nur Class-Dateien, sondern auch andere © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 68 / 409 Dinge erzeugt werden, wie z.B. Jar-Dateien – und sei es nur aus Gründen der Testbarkeit. Dann wird diese Vorgehensweise – jedes Package händisch auf der Kommandozeile zu übersetzen – viel zu arbeits- und fehlerintensiv. Eine Lösung ist natürlich die Verwendung einer IDE wie z.B. Eclipse, in der alle diese Dinge mit einem hohen Grad an Automation erledigt werden können. Aber möglicherweise wollen oder können Sie keine IDE verwenden. Oder – und das spiegelt die Praxis in typischen Projekten eher wider – Sie müssen neben der IDE auch noch eine Kommandozeilen-Lösung unterstützen20. In diesem Fall ist das typische Werkzeug im Java Umfeld „Ant“. Die fleissige Ameise Ant ist ein in Java geschriebenes Open-Source Tool21, mit dem komplette Builds beschrieben und automatisiert durchgeführt werden können – solange auf dem Build-System eine JVM zur Verfügung steht. Für Details und weitere Informationen sei hier auf das Internet verwiesen – siehe z.B. http://jakarta.apache.org/ant. Ein zweites im Java Umfeld sehr verbreitetes Build-Tool ist Maven (http://maven.apache.org/ ), aber es existieren noch viele weitere Build-Tools. Zusätzlich empfehle ich Ihnen neben einem Build-Tool auf jeden Fall eine Versions-Verwaltung für Ihre Quelltexte – VCS – Version-Control-Systems. Leider sprengen detailierte Informationen zu VCS den Rahmen des Java Tutorials, lesen Sie z.B. in Wikipedia nach: http://de.wikipedia.org/wiki/Versionsverwaltung. Folgende Versions-Verwaltungen sind die wohl zur Zeit verbreitesten Systeme: • CVS http://savannah.nongnu.org/projects/cvs • Subversion – SVN http://subversion.apache.org/ • Mercurial http://mercurial.selenic.com/ • Git http://git-scm.com/ • Perforce http://www.perforce.com/ 4.5 Source-Path und Class-Path 4.5.1 Source-Path Was sollen eigentlich beim Compilieren und Ausführen diese komischen Source- und ClassZ.B. für automatische Builds jede Nacht, oder für Builds auf zentralen Servern für die keine IDE vorhanden ist. 21 Ant ist ein Teil des Jakarta Projekts. 20 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 69 / 409 Path Angaben. Schauen wir uns dafür mal den Source-Path beim Compilieren an, und nehmen wir an, wir haben in einem Projekt die zwei Klassen „Class1“ und „Class2“ in zwei parallel liegenden Packages „pack1“ und „pack2“. pack1 pack2 Class1 Class2 Abb. 4-35 : 2 Klassen in zwei parallelen Packages Selbst wenn wir noch nicht viel über Klassen und Packages wissen, sollte uns klar sein, dass es passieren kann (und auch soll) dass eine Klasse die andere nutzt. package pack1; public class Class1 { private pack2.Class2 cl; } // Class1 nutzt Class2 aus dem Package pack2 Klar sollte uns auch sein, dass der Compiler überprüft, ob es die Klasse z.B. überhaupt gibt, ob „Class1“ sie benutzen darf, oder ob die Benutzung korrekt erfolgt (z.B. entsprechende Funktionen in der Klasse vorhanden sind)22. Woher weiss der Compiler aber nun, wo die Klasse „pack2.Class2“ zu finden ist, um diese Dinge zu überprüfen? Hier kommt drei Dinge ins Spiel: 1) Eine Klasse heißt wie die Datei, in der sie steht. 2) Ein Package heißt wie das Verzeichnis, das es darstellt. 3) Der Source-Path zeigt auf die Wurzel(n) der Package-Struktur. pack1 Class-Path == d:\source d:\ source pack1 Class1.java pack2 Class2.java Class1 pack2 Class2 Abb. 4-36 : Klassen, Dateien, Packages, Verzeichnisse und der Source-Path Der Compiler findet im Quelltext also die Referenzierung der Klasse „pack2.Class2“ und weiß In Java werden solche Dinge zum Compile-Zeitpunkt vom Compiler gecheckt. Damit können Fehler schon zur Compile-Zeit gefunden werden, die ansonsten zu Fehlern zur Laufzeit führen. Es gibt Sprachen, die dies nicht so machen – und die Fehler für den Benutzer aufsparen. 22 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 70 / 409 sofort folgende Dinge: 1) Das Package heißt „pack2“ – also muss es im Source-Path Verzeichnis als der Wurzel aller Sourcen ein Verzeichnis mit Namen „pack2“ geben. 2) Die Klasse heißt „Class2“, also muss es in diesem Verzeichnis eine Datei „Class2.java“ geben, die die Klasse „Class2“ enthält und detailiert definiert. Mit diesem Wissen kann er also die Definition der Klasse „Class2“ finden, sie einlesen, und alle notwendigen Überprüfungen durchführen. Hinweis – in Wirklichkeit kann da noch ein bisschen mehr passieren, da z.B. der Source-Path mehrere Verzeichnisse umfassen kann, oder der Compiler auch mit Class-Dateien und JarFiles umgehen kann. Aber auch da passiert im Prinzip genau das gleiche wie hier beschrieben. Der Source-Path hilft dem Compiler also referenzierte Klassen zu finden, um seinen CompileJob mit möglichst vielen Überprüfung durchführen zu können. 4.5.2 Class-Path Und was ist mit dem Class-Path? Das ist genau das gleiche für die virtuelle Maschine. Der Class-Path beschreibt, wo der Byte-Code der Klassen zu finden ist. D.h. referenziert er die Wurzel-Verzeichnisse für die Class-Dateien. Und daher müssen auch die Class-Dateien in korrespondierenden Verzeichnissen abgelegt werden, und haben auch den Namen der Klasse als Datei-Namen. Wird kein Class-Path angeben, wird das aktuelle Verzeichnis als Class-Path angenommen. Folgende Aufrufe sind also identisch: > java KlassenName > java –cp . KlassenName Der Class-Path kann aber auch über die Environment Variable „CLASSPATH“ einen Default bekommen, der dann statt des aktuellen Verzeichnisses genommen wird. In so einem Fall wären die beiden obigen Aufrufe nicht identisch, und der Class-Path müßte auch bei ClassDateien im aktuellen Verzeichnis angegeben werden. 4.6 Eclipse Viel einfacher wird die Java Entwicklung mit einer IDE (Integrated Development Environment), da dort alle Tools wie Editor, Compiler, JVM und viele weitere unter einer grafischen Oberfläche zusammengeführt sind. Besonders komfortabel wird die Entwicklung, wenn ein wahres Monster von IDE wie z.B. Eclipse (siehe Kapitel 4.1.3) genutzt wird. Aber – seien Sie gewarnt – mit dem Komfort kommt Hand-in-Hand auch eine gewisse Komplexität. Denn Eclipse kann super-viel, und hat auch seine eigene Benutzungs-Philosophie. Dieses Kapitel soll eine einfache kleine Einführung in Eclipse sein. Ihnen sollte aber klar sein, © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 71 / 409 dass es nur die rudimentärtsten Features von Eclipse erklären kann23. Achtung – alle Screen-Shots in diesem Eclipse-Einführungs-Kapitel sind unter Microsoft Windows 7 im klassischen Design entstanden. Je nach dem von Ihnen verwendeten Betriebssystem und Design kann das Aussehen bei Ihnen von den Screen-Shots abweichen. Hinzu kommt noch, dass sich auch das Erscheinungsbild der Eclipse selber und aller Perspektiven in einem hohen Maße angepaßen und individualisieren werden. Es kann also ohne weiteres sein, dass Ihre konkrete Eclipse-Installation bei Ihnen ganz anders aussieht. 4.6.1 Workspaces Nach dem Start von Eclipse müssen Sie als erstes einen Workspace auswählen bzw. neu anlegen. In Eclipse sind Workspaces für die globale Strukturierung von Projekten da – d.h. in ihnen können mehrere Projekte zusammengefaßt werden24. Ein Workspace entspricht einem Verzeichnis, in dem nachher alle Teile des Workspaces liegen. Wenn Sie einen neuen Workspace anlegen wollen, müssen Sie im Workspace-Launcher daher nur ein entsprechendes Verzeichnis angeben – hier im Beispiel „e:\java\bsp-workspace“. Abb. 4-37 : Eclipse Workspace Launcher Eclipse startet dann mit seinem Begrüßungs-Bildschirm, den wir jetzt ignorieren wollen – d.h. wir schließen ihn einfach. Später können Sie über den Begrüßungs-Bildschirm z.B. Tutorials und Beispiele aus dem Internet öffnen. Es gibt dicke Bücher nur zum Thema Eclipse – darin finden sie alles wichtige, In der Praxis ist dies sehr normal, dass ein Projekt aus mehreren kleinen Projekten besteht. Man macht dies, um mehr Übersichtlichkeit zu erreichen. 23 24 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 72 / 409 Abb. 4-38 : Eclipse Begrüßungs Bildschirm Abb. 4-39 : Eclipse Begrüßungs Bildschirm schließen Danach sehen wir unseren neuen (noch leeren) Workspace in der Java Perspektive. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 73 / 409 Abb. 4-40 : Leerer Workspace in der Java Perspektive 4.6.2 Projekt Um in der Eclipse ein Java Programm zu schreiben – und sei es noch so klein – muß ein JavaProjekt vorhanden sein. Dies muss also als erstes angelegt werden. Möglich ist dies über viele alternative Wege, z.B. über das Menü mit „File => New => Java Project...“, über die Toolbar oder über das Kontext-Menü im Package-Explorer auf der linken Seite, den man im folgenden Screen-Shot sieht: © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 74 / 409 Abb. 4-41 : Neues Projekt anlegen Es erscheint ein Wizard, der Sie durch das Anlegen eines neues Projekts führt. Auf der ersten Seite müssen Sie dem Projekt einen Namen geben, hier im Beispiel „Hallo Welt“, da wir ein Hallo-Welt Programm implementieren wollen. Man kann hier auf der ersten Seite und der folgende Seite des Wizards noch weitere Voreinstellungen vornehmen – aber sie interessieren uns zur Zeit nicht, da wir nur ein ganz einfaches Java-Projekt anlegen wollen. Also beenden Sie den Wizard mit „Finish“. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 75 / 409 Abb. 4-42 : Projekt-Wizard Nach erfolgreichem Beenden des Projekt-Wizards sehen Sie folgende Workspace Ansicht mit unserem „Hallo Welt“ Projekt. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 76 / 409 Abb. 4-43 : Workspace mit neuem „Hallo Welt“ Projekt 4.6.3 Klasse Als nächstes müssen wir eine Klasse erzeugen, denn Klassen sind ja das Strukturierungsmittel von Java, und ohne eine Klasse läuft gar nichts – siehe Kapitel 3.6. Dafür haben wir wieder mehrere Möglichkeiten, z.B. über das File-Menü, mit dem Toolbar-Button (siehe rote Markierung im folgenden Screen-Shot) – oder mit dem Kontext-Menü auf dem Source-Folder des Projekts mit „New => Class“, wie im Screen-Shot zu sehen: © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 77 / 409 Abb. 4-44 : Neue Klasse anlegen Dann öffnet sich der Klassen-Wizard der Eclipse. Primär muß hier der Name der Klasse und das Package angegeben werden: • In unserem Beispiel nenne ich die Klasse „HelloWorldAppl“ – siehe zweiter roter Kasten im folgenden Screen-Shot. • Außerdem lege ich die Klasse in das Package „examplepackage“ – siehe erster roter Kasten im folgenden Screen-Shot – denn in realen Programmen sollten alle Klassen Packages zugeordnet sein, vergleiche Kapitel 3.8 und Kapitel 13.5. • Eine weiteres Feature des Klassen-Wizards, das wir direkt nutzen wollen, betrifft die Möglichkeit sich automatisch eine Main-Funktion generieren zu lassen. Das wollen wir nutzen, und selektieren die entsprechende Checkbox daher – siehe dritter roter Kasten © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 78 / 409 Abb. 4-45 : Klassen-Wizard Die Namen „HelloWorldAppl“ und „examplepackage“ entsprechen den Java-Konventionen, die wir in Kapitel 3.2.5 kennen gelernt haben: • Klassen-Namen beginnen groß und werden kapitalisiert geschrieben • Package-Namen werden durchgängig klein geschrieben Auch bei der Einhaltung solcher Konventionen unterstützt uns die Eclipse. Im oberen Bereich des Wizards wird eine Warnung eingeblendet, wenn wir solche oder andere Konventionen verletzen. Im folgenden Screen-Shot z.B. sehen wir eine Warnung, wenn wir kein Package, d.h. das sogenannte Default-Package, verwenden: © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 79 / 409 Abb. 4-46 : Warnung bei Nutzung des Default-Packages Oder im folgenden Beispiel sehen wir eine Warnung, da die Konvention für Klassen-Namen verletzt wurde – hier z.B. beginnt der Klassen-Name mit einem Kleinbuchstaben: Abb. 4-47 : Warnung bei fehlerhaftem Klassen-Namen Alle weiteren Features des Klassen-Wizards ignorieren wir erstmal. Zum einen sagen uns viele noch nichts, zum anderen müssen Sie ja noch was zum Spielen und Ausprobieren haben. Mit „Finish“ beenden Sie den Wizard, und Eclipse generiert nach Ihren Angaben: • Ein Package, wenn angeben, und natürlich inkl. Verzeichnis im Workspace Verzeichnis auf der Festplatte. • Eine Klasse in der korrekten Datei, abgelegt im richtigen Package (Verzeichnis). • Und eine Main-Funktion in der Klasse. So sieht es das alles dann im Workspace aus. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 80 / 409 Abb. 4-48 : Neue Klasse „HalloWeltAppl“ im Workspace Ich will auch hier gar nicht auf alle Features von Eclipse eingehen – genau genommen nicht mal einen Bruchteil – aber ein paar Dinge möchte ich erklären. Den Rest überlasse ich Ihrem Forschergeist. • In der Mitte ist der sogenannte Editor-Bereich, in dem die Quelltexte angezeigt und von Ihnen editiert werden können. Der Editor unterstützt sie mit einer Unmenge von Features, so z.B. Code-Completion, automatischen Builds, Quick-Fixes, Mark-Occurrences, Folding, Syntax-Highligthing, Refactoring, uvm. Viel Spaß beim Forschen und Ausprobieren. • Links im Package-Explorer haben Sie eine logische Sicht auf Ihren Workspace und Ihre Projekte. Hier finden Sie alle Projekte, Package, Klassen, Libraries und all die anderen Dinge, die in einem typischen Projekt vorkommen, wieder. • Unten finden Sie mehrere Views, von denen der Problems-View vorne liegt. In ihm werden die Compiler-Fehler und –Warnungen des Codes angezeigt. Im Hintergrund liegen Views, die die Java-Doc bzw. die Deklaration des jeweils selektierten Elements anzeigen. In diesem Bereich wird sich später auch der Consolen-View öffnen. • Rechts finden sich mehrere Views untereinander. Da der Screen-Shot so klein ist, sind sie kaum zu erkennen und so auch nicht zu verwenden. Ich hoffe, Sie haben einen größeren © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 81 / 409 Bildschirm: - Ganz oben findet sich der Task-Views, der Aufgaben verwalten kann. - Darunter liegt der sogenannte Outline-View, der eine logische Sicht der aktuellen Datei ermöglicht. D.h. hier finden Sie alle Imports, Klassen, Konstruktoren, Funktionen und Attribute wieder. Per Doppelklick können Sie sie direkt anspringen. Dies ist das Default-Aussehen der Eclipse 4.4 – Sie können es aber in großem Umfang ändern und an Ihre Bedürfnisse und Vorlieben anpassen. All diese Features sollen uns aber nicht davon abhalten zu programmieren – darum geht es uns ja gerade. Und dazu gehört natürlich auch unser Hallo-Welt Programm fertig zu programmieren. All die Features helfen Ihnen hier zwar – aber denken und tippen müssen Sie immer noch selber. Ich habe es mal gemacht – und das Ergebnis sehen Sie in der nächsten Abbildung – unsere Main-Funktion enthält nun eine Anweisung, um „Hallo Welt“ auszugeben: Abb. 4-49 : Fertiger Quelltext im Wizard 4.6.4 Start und Ab... Jetzt muss unser ultimatives Programm nur noch laufen, und wir können ans Verkaufen und © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 82 / 409 Geld verdienen gehen25. Es gibt viele Möglichkeiten unser Programm zu starten. Und manche davon sind kompliziert, da man viele Einfluss-Möglichkeiten hat (z.B. könnte man dem Programm mal ein anderes JRE unterschieben, um seine Kompatibilität nach unten oder oben zu checken). Mit einer der einfachsten Arten das Programm zu starten ist die Klasse mit der Main-Funktion im Package-Explorer zu selektieren, und dann über das Kontext-Menü „Run As => Java Application“ das Programm direkt von hier zu starten. Falls Sie diese Vision glauben, sind sie a) sehr leichtgläubig und haben b) das Tutorial nicht ordentlich gelesen. Denn in Kapitel 1 steht extra, dass man nicht alles glauben soll was geschrieben steht – erst recht nicht, wenn es hier im Tutorial steht ;-). 25 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 83 / 409 Abb. 4-50 : Starten des Programms Unser Programm ist ein reines Kommandozeilen-Programm ohne GUI. Eclipse öffnet jetzt keine Kommandozeile, sondern stellt einen Consolen-View zur Verfügung, in den das Programm seine Ausgaben (und später auch Eingaben) macht. Abb. 4-51 : Ausgabe des Programms 4.6.5 Warnungen Der Java-Compiler in der Eclipse wird in der Default-Einstellung bei einigen Beispielen aus dem Tutorial Warnungen erzeugen. Im Prinzip sind diese Warnungen gut und sehr hilfreich, und wir sollten Code schreiben der keine Warnungen produziert. Aber bei unseren Beispielen machen wir aber ab und zu Dinge absichtlich falsch, um etwas zu lernen – dann sind uns diese Warnungen egal. Wenn Ihr Workspace aber viele Projekte enthält, in denen – aus welchen Gründen auch immer – zum Teil solche Warnungen auftauchen, dann können eine Menge Warnungen zusammenkommen. In einem solchen Fall gehen dann schnell wichtige Warnungen gegenüber den „fehlerhaften“ Warnungen unter und fallen nicht auf. In einem solchen Fall sollte man die „fehlerhaften“ Warnungen deaktivieren. Sie haben hierzu zwei Möglichkeiten: • Sie können die Warnung selektiv im Quelltext mit Annotations (siehe Kapitel todo) © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 84 / 409 deaktivieren. Dies ist die meistens zweckmäßige Variante, wenn die Warnung an sich sinnvoll ist – und nur an dieser einen Stelle nicht. • Oder Sie deaktivieren die Warnung global in der Eclipse in den Eclipse Preferences – siehe nächste Screenshots. Damit wird die Warnung natürlich nirgendswo mehr erzeugt, und möglicherweise verhindern Sie damit wichtige Hinweise des Compilers auf fehlerhaften Code. Hier zwei Beispiele von Code-Features, die Warnungen erzeugen und in den Beispielen des Tutorials auftauchen können: • Das Schreiben serialisierbarer Klassen ohne „serialVersionUID“ Attribute – siehe Kapitel todo • Die Verwendung typloser Container – siehe Kapitel todo Möchten Sie diese Warnungen unterdrücken, so empfehlen sich selektive Annotations (siehe oben). Alternativ können Sie die Warnungen in den Eclipse Preferences einfach abstellen. Die Eclipse Preferences erreichen Sie über das Menü „Window => Preferences...“. Unter „Java => Compiler => Errors/Warnings“ finden Sie die Details zu den Eclipse/Java Warnungen und Fehlern – hier ändern Sie dann die Einstellungen von „Warning“ auf „Ignore“ – und schon haben Sie Ihre Ruhe. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 85 / 409 Abb. 4-52 : Eclipse Warnungs Preferences für „serialVersionUID“ Abb. 4-53 : Eclipse Warnungs Preferences für „typlose Container“ 4.6.6 Eclipse Okay, damit ist unser Kurz-Trip durch die Eclipse fast beendet. Wir haben kurz die wichtigsten Features für die erste Entwicklung kennen gelernt, wie Workspaces, Projekte, Klassen und den Consolen-View. Im nächsten Kapitel todo werden wir noch sehen, wie man fertige Projekt und Einstellungen importiert. Alles weitere kommt im Laufe der Zeit von alleine Aber vergessen Sie nicht bei all dem Spielen in und mit Eclipse. Eclipse ist nur ein Werkzeug, eine IDE – das Kern-Thema der Vorlesung ist und bleibt Java. 4.7 Projekte importieren Damit Sie nicht alle Beispiele abtippen müssen, finden Sie auf meiner Homepage unter © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 86 / 409 http://www.wilkening-online.de/tutorial-java.html nicht nur das jeweils neuste Java-Tutorial, sondern auch viele der Beispiele und Musterlösungen des Tutorials als Eclipse Workspace. Laden Sie die entsprechende 7z Datei „JavaTutorialWorkspace.7z“ herunter und packen Sie sie aus. Ich habe das mal beispielhaft direkt unter „E:\“ gemacht. Abb. 4-54 : Ausgepackter Tutorial Workspace auf “E:\” Dann starten Sie die Eclipse und legen einen neuen Workspace an – ich habe z.B. „e:\java\tutorial-wsp“ genommen. Natürlich müssen Sie keinen neuen Eclipse Workspace anlegen, sondern können die Beispiele auch in einen vorhandenen Eclipse Workspace übernehmen – ich habe hier aus Gründen der Übersichtlichkeit einen Neuen gewählt. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 87 / 409 Abb. 4-55 : Neuer Eclipse Workspace für die Tutorial Beispiele Öffnen Sie im Package-Explorer das Kontext-Menü (z.B. mit der rechten Maustaste) und wählen Sie den Menü-Eintrag „Import“ aus. Abb. 4-56 : Kontext-Menü mit Import Eintrag Dann öffnet sich der Import Wizard. Wählen Sie unter „General“ den Eintrag „Existing Projects into Workspace“ aus und drücken dann den Button „Next“. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 88 / 409 Abb. 4-57 : Import Wizard Seite 1 Damit wechseln Sie zur Seite 2 des Import Wizards: • Im oberen Bereich geben Sie an, wo der ausgepackte Workspace liegt – in meinem Fall „E:\JavaTutorialWorkspace“ • Nach Eingabe des Verzeichnisses werden im mittleren Projekt Bereich alle Eclipse-Projekte aufgelistet, die die Eclipse in dem angegebenen Verzeichnis gefunden hat. Hier können Sie selektiv die Projekte auswählen, die Sie importieren möchten – defaultmäßig sind hier alle Projekte selektiert. • Wenn Sie die Quelltexte in Ihren neuen Workspace kopieren möchten, dann müssen Sie noch die Checkbox bei „Copy projects into workspace“ setzen. Falls Sie dies nicht machen, verlinkt die Eclipse Ihre neuen Projekte in Ihrem neuen Workspace mit den Dateien im Tutorial-Verzeichnis. Dann dürfen Sie das Verzeichnis natürlich nicht löschen oder anderweitig verändern. Darum kopiere ich die Dateien hier lieber in meinen neuen Workspace. Je nach Workflow und Anforderung kann natürlich auch das Verlinken in der Praxis sehr sinnvoll und hilfreich sein. • Danach müssen Sie nur noch den Button „Finish“ drücken. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 89 / 409 Abb. 4-58 : Import Wizard Seite 2 Damit stehen Ihnen dann alle Beispiele und Musterlösungen in Ihrem Workspace zur Verfügung. Viel Spaß beim Ausführen, Analysieren, Verstehen und Verändern. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 90 / 409 Abb. 4-59 : Der fertig importierte Eclipse Workspace 4.8 Aufgaben Der Sinn dieser ersten Aufgaben besteht primär darin, sich mit den Entwicklungs-Tools in Form von Editor, Java-Compiler „javac“, JVM „java“, und Ihrer ausgewählten IDE wie z.B. Eclipse oder NetBeans vertraut zu machen. Sekundär werden damit natürlich auch die ersten SprachElemente geübt. Achten Sie bitte auch bei diesen ersten Programm schon auf sinnvolle Namen und eine vernünftige Einrückung. Und das gilt natürlich erst Recht für alle weiteren Aufgaben. 4.8.1 Aufgabe „Hallo Welt“ Schreiben Sie eine erste Klasse ohne Package mit Main-Funktion, in der „Hallo Welt“ auf der Kommandozeile ausgegeben wird. 1) Entwickeln und testen Sie das Programm mit einem Editor, dem Java Compiler „javac“, und mit der JVM „java“ in einer Kommandozeile. 2) Entwickeln und testen Sie das Programm in Ihrer IDE wie z.B. Eclipse oder NetBeans. Lösung siehe Kapitel 4.9. 4.8.2 Aufgabe „GUI-Fenster“ Schreiben Sie eine Klasse ohne Package mit Main-Funktion, in der ein GUI Fenster geöffnet wird. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 91 / 409 1) Entwickeln und testen Sie das Programm mit einem Editor, dem Java Compiler „javac“, und mit der JVM „java“ in einer Kommandozeile. 2) Entwickeln und testen Sie das Programm in Ihrer IDE wie z.B. Eclipse oder NetBeans. Lösung siehe Kapitel 4.10. 4.8.3 Aufgabe „Package“ Schreiben Sie eine Klasse in einem Package mit Main-Funktion, in der etwas passendes auf der Kommandozeile ausgegeben wird. 1) Entwickeln und testen Sie das Programm mit einem Editor, dem Java Compiler „javac“, und mit der JVM „java“ in einer Kommandozeile. 2) Entwickeln und testen Sie das Programm in Ihrer IDE wie z.B. Eclipse oder NetBeans. Lösung siehe Kapitel 4.11. 4.8.4 Aufgabe „Packages“ Schreiben Sie eine Klasse in einem verschachteltem Package (d.h. das Package liegt wieder in einem Package) mit Main-Funktion, in der etwas passendes auf der Kommandozeile ausgegeben wird. 1) Entwickeln und testen Sie das Programm mit einem Editor, dem Java Compiler „javac“, und mit der JVM „java“ in einer Kommandozeile. 2) Entwickeln und testen Sie das Programm in Ihrer IDE wie z.B. Eclipse oder NetBeans. Lösung siehe Kapitel 4.12. 4.8.5 Aufgabe „Summe“ Schreiben Sie ein Programm, dass zwei Gleitkomma-Zahlen von der Kommandozeile einliest und ihre Summe ausgibt. Fangen sie fehlerhafte Eingaben ab, und geben sie im Falle eines Fehlers eine Meldung aus. 1) Entwickeln und testen Sie das Programm mit einem Editor, dem Java Compiler „javac“, und mit der JVM „java“ in einer Kommandozeile. 2) Entwickeln und testen Sie das Programm in Ihrer IDE wie z.B. Eclipse oder NetBeans. Dieses Programm hat – bezogen auf unser aktuelles Wissen – einen Knackpunkt: das Einlesen von Kommandozeile (siehe Kapitel 3.10) ist relativ aufwändig und setzt eine Menge Pragmatismus voraus26. Etwas einfacher wäre die Benutzung von KommandozeilenArgumenten (siehe Kapitel 3.5). Pragmatismus meint hier: ich übernehme den Code und passe ihn an meine Bedürfnisse an, ohne ihn auch nur annähernd zu verstehen. 26 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 92 / 409 Nun werden viele der noch folgenden Programme so viel Benutzer-Interaktion erfordern, dass Kommandozeilen-Argumente keine Lösung darstellen, aber für diese Aufgabe (und noch einige folgende) wäre dies eine akzeptable Lösung. Und später haben wir ja mehr Erfahrung und mehr Wissen – bis dahin erscheint uns die Eingabe vielleicht gar nicht mehr so mystisch. Nachteilig an der Kommandozeilen-Argument Lösung ist aber, dass die Änderung der Kommandozeilen-Argumente in Eclipse aufwändiger ist. Für das Entwickeln und Testen in Eclipse wäre das Einlesen von Kommandozeile viel angenehmer und flexibler. Von daher schlage ich folgenden Kompromiß vor: • Für den ersten Wurf mit dem Java-Compiler „javac“ und der JVM „java“ implementieren sie die Aufgabe mit Kommandozeilen-Argumenten. • Und im zweiten Wurf mit Eclipse – und dann ja auch mit viel mehr Erfahrung – lesen Sie die Summanden von der Kommandozeile ein. Lösung siehe Kapitel 4.13. 4.8.6 Aufgabe „Ausgabe Verzeichnis“ Compilieren Sie die Programme der fünf vorherigen Aufgaben mit dem Java Compiler „javac“ so, dass der Byte-Code in einem eigenen Verzeichnis liegt. Starten Sie die Programme danach mit der JVM „java“ in einer Kommandozeile. Denken Sie sich einen sinnvollen Ausgabe-Pfad Namen aus. Lösung siehe Kapitel 4.14. 4.9 Lsg. zu Aufgabe „Hallo Welt“ – Kap. 4.8.1 Diese Aufgabe – wie eigentlich fast alle in diesem Kapitel – ist von der Java Seite her sehr einfach, da der Quelltext z.B. in den Kapiteln 3.1.1 oder 4.3.1.1 oder 4.6.3 fertig vorliegt. Der Vollständigkeit halber sei hier aber noch mal aufgeführt. public class Kap0410LsgHalloWelt { public static void main(String[] args) { System.out.println("Hallo Welt"); } } Für den ersten Teil der Aufgabe, müssen nur noch der Java-Compiler „javac“ und die JVM „java“, wie in Kapitel 4.3.1.1 beschrieben, benutzt werden. Die Benutzung der IDE Eclipse wird in Kapitel 4.6 beschrieben. Dies auszuführen ist dann der zweite Teil der Aufgabe. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 93 / 409 4.10 Lsg. zu Aufgabe „GUI-Fenster“ – Kap. 4.8.2 Auch diese Aufgabe ist ein Übernehmen des fertigen Quelltextes aus z.B. Kapitel 3.1.2, und ein Anwenden der Tools „javac“ und „java“ wie in Kapitel 4.3.1.2 beschrieben, bzw. von Eclipse (siehe Kapitel 4.6). import javax.swing.JFrame; public class Kap0411LsgGui { public static void main(String[] args) { JFrame frame = new JFrame("Mein erstes GUI Fenster"); frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); frame.setLocation(200, 200); frame.setSize(300, 100); frame.setVisible(true); } } 4.11 Lsg. zu Aufgabe „Package“ – Kap. 4.8.3 Von der Seite der Programmierung ist diese Aufgabe im Prinzip wieder ein „Hallo Welt“. Einziger Unterschied ist, dass die Klasse in einem Package stehen soll, und daher der Quelltext mit einer entsprechenden Package-Anweisung beginnen muss – siehe Kapitel 3.8. Er könnte also z.B. so aussehen: package mypackage; public class Appl { public static void main(String[] args) { System.out.println("Ausgabe aus einer Klasse in einem Package"); } } Die eigentliche Schwierigkeit der Aufgabe liegt im Bereich der Daten-Organisation und der Tools: • Bei Verwendung von der Tools „javac“ und „java“ müssen Sie dafür sorgen, dass der Quelltext in einem entsprechenden Verzeichnis liegt. Und Sie müssen das Compilieren und Starten des Programms entsprechend Kapitel 4.3.2.2 durchführen. • Mit Eclipse ist das ganze viel einfacher. Bei der Erzeugung der Klasse mit dem KlassenWizard (siehe Kapitel 4.6.3) müssen Sie einfach im Package-Feld den gewünschten Package-Namen angeben: alles andere erledigt Eclipse für Sie automatisch richtig. 4.12 Lsg. zu Aufgabe „Packages“ – Kap. 4.8.4 Diese Aufgabe unterscheidet sich von Aufgabe 4.8.3 (Lösung Kapitel 4.11) nur durch drei verschachtelte statt einem einzigen Package. Der Quelltext ist also bis auf die angepaßte Package Anweisung und die angepaßte Ausgabe identisch zu Kapitel 4.11. package aussen.mitte.innen; © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 94 / 409 public class Appl { public static void main(String[] args) { System.out.println("Ausgabe aus einer Klasse in drei Packages"); } } Und was ist sonst zu tun? • Mit den Kommandozeilen-Tools ist händisch für die richtige Verzeichnis-Struktur zu sorgen. Danach wieder die Benutzung vergleichbar zur Lösung 4.11. • In Eclipse müssen im Klassen-Wizard nur alle Packages angegeben werden – den Rest erledigt Eclipse für uns ☺. 4.13 Lsg. zu Aufgabe „Summe“ – Kap. 4.8.5 4.13.1 Mit Kommandozeilen Argumenten 1 Die erste Lösung basiert auf der Auswertung der Kommandozeilen Argumente, statt die Zahlen von der Kommandozeile interaktiv einzulesen. Dieses Programm läßt sich mit etwas Pragmatismus bei der Anwendung der Sprach-Elemente aus den Kapiteln 3.5 („Arrays und Kommandozeilen-Argumente“), 3.9 („Exceptions“) und 3.11 („Konvertierungen“) recht einfach implementieren. Für die Addition benötigen wir zwei Summanden, d.h. müssen auch zwei Kommandozeilen Argumente übergeben werden. Ist dies nicht der Fall, geben wir eine Fehlermeldung aus, und beenden das Programm. public static void main(String[] args) { if (args.length!=2) { System.out.println("Das Programm erwartet zwei Zahlen als Eingabe"); return; } ... } Dann müssen die Operanden, die ja als Strings vorliegen, in Double Werte gewandelt werden. Da dies schief gehen kann – der Benutzer könnte ja z.B. Texte statt Zahlen übergeben, könnte die Wandlungs-Funktion „Double.parseDouble“ eine Exception werfen. Diese fangen wir ab, melden dann einen Fehler, und beenden das Programm. double d1; try { d1 = Double.parseDouble(args[0]); } catch (Exception x) { System.out.println("Der erste Parameter ist keine Gleitkomma-Zahl"); return; } Und da wir zwei Summanden haben, benötigen wir den Code zweimal in fast identischer Weise. Unterschiede finden sich in der Double-Variablen „d1“ bzw. „d2“, dem Kommandozeilen Argument „args[0]“ bzw. „args[1]“ und der Fehlermeldung. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 95 / 409 Am Schluss müssen die zwei Zahlen nur noch addiert werden, und die Summe muss mit einer Meldung ausgegeben werden. double sum = d1 + d2; System.out.println(d1 + " + " + d2 + " = " + sum); Alles zusammen ergibt dann folgendes Programm: public class Appl { public static void main(String[] args) { if (args.length!=2) { System.out.println("Das Programm erwartet zwei Zahlen als Eingabe"); return; } double d1; try { d1 = Double.parseDouble(args[0]); } catch (Exception x) { System.out.println("Der erste Parameter ist keine Gleitkomma-Zahl"); return; } double d2; try { d2 = Double.parseDouble(args[1]); } catch (Exception x) { System.out.println("Der zweite Parameter ist keine Gleitkomma-Zahl"); return; } double sum = d1 + d2; System.out.println(d1 + " + " + d2 + " = " + sum); } } 4.13.2 Mit Kommandozeilen Argumenten 2 Wenn man nicht so viel Werte auf super-detailierte Fehler-Meldungen legt, kann man Lösung 1 noch vereinfachen. In Kapitel 3.9 wurde schon erwähnt, dass Probleme in Java durch Exceptions gemeldet werden – und dazu gehört auch der Zugriff auf nicht existente ArrayElemente (Kapitel 3.5). Mit dieser Erkenntnis kann man sich das Leben (das Programm) vereinfachen, und die erste Fehlerabfrage auf die Existenz der beiden Kommandozeilen Argumente sparen. Man greift beim Konvertieren einfach blind ins Array, und existiert das Element nicht, so wird auch dies durch eine Exception gemeldet. Wir sollten vielleicht nur die Fehler-Meldung etwas anpassen, und schon sieht die Konvertierung so aus: double d1; try { d1 = Double.parseDouble(args[0]); } catch (Exception x) { System.out.println("Probleme mit dem ersten Parameter"); return; } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 96 / 409 Und das ganze Programm ist im Prinzip nur eine Wiederholung dieses Blocks mit Ausgabe der Summe und dem notwendigen Drum-Herum (Klasse mit Main-Funktion). public class Appl { public static void main(String[] args) { double d1; try { d1 = Double.parseDouble(args[0]); } catch (Exception x) { System.out.println("Probleme mit dem ersten Parameter"); return; } double d2; try { d2 = Double.parseDouble(args[1]); } catch (Exception x) { System.out.println("Probleme mit dem zweiten Parameter"); return; } double sum = d1 + d2; System.out.println(d1 + " + " + d2 + " = " + sum); } } 4.13.3 Einlesen von Kommandozeile Auch die Version mit dem interaktiven Einlesen von Kommandozeile ist nicht wirklich schwieriger. Zuerst muss – wie in Kapitel 3.10 beschrieben – die Tastatur „System.in“ in einem „InputStreamReader“ und einem „BufferedReader“ gewrappt werden. InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); Achtung - vergessen Sie die beiden Import-Anweisungen nicht. Statt jetzt die Kommandozeilen Argumente auszuwerten, müssen Sie einfach nur eine Zeile als String von der Kommandozeile einlesen: • Am besten vorher mit einer sinnvollen Aufforderung für den Benutzer. • Da das Einlesen schief gehen kann, kann „reader.readLine()“ eine Exception werfen. Da dies eine Checked-Exception ist, muß diese Zeile in einem Try-Catch-Block stehen. Aber den haben wir ja schon wegen der Konvertierung „String-zu-Double“. • Da jetzt sowohl die Eingabe als auch die Konvertierung in ein und dem gleich Try-CatchBlock stehen, kann nicht mehr unterschieden werden, ob das Problem beim Einlesen entstanden ist, oder bei der Konvertierung. Unsere Fehlermeldung sollte also recht allgemein gehalten sein.27 double d1; try { System.out.print("Bitte geben Sie den ersten Summanden ein: "); String in = reader.readLine(); Wer gerne bessere Fehlermeldungen hätte, muss die Eingabe und die Konvertierung einzeln mit einem Try-Catch-Block wrappen, und kann dann dediziert reagieren. 27 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 97 / 409 d1 = Double.parseDouble(in); } catch (Exception x) { System.out.println("Probleme mit der Eingabe"); return; } Das ganze zweimal, und dann wie gehabt die Summe bilden und ausgeben. Dann noch das Rum-Herum von Klasse und Main-Funktion, und fertig ist das Programm mit Einlesen von der Kommandozeile. Es hat doch gar nicht weg getan, oder? import java.io.InputStreamReader; import java.io.BufferedReader; public class Appl { public static void main(String[] args) { InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); double d1; try { System.out.print("Bitte geben Sie den ersten Summanden ein: "); String in = reader.readLine(); d1 = Double.parseDouble(in); } catch (Exception x) { System.out.println("Probleme mit der Eingabe"); return; } double d2; try { System.out.print("Bitte geben Sie den zweiten Summanden ein: "); String in = reader.readLine(); d2 = Double.parseDouble(in); } catch (Exception x) { System.out.println("Probleme mit der Eingabe"); return; } double sum = d1 + d2; System.out.println(d1 + " + " + d2 + " = " + sum); } } 4.14 Lsg. zu Aufgabe „Ausgabe Verzeichnis“ – Kap. 4.8.6 Zu dieser Lösung gibt es nicht viel zu schreiben, da alle Quelltexte schon in den LösungsKapiteln zuvor entwickelt worden sind. Hier sollen ja nur beim Compilieren mit dem JavaCompiler „javac“ der Byte-Code, d.h. die Class-Dateien, in ein extra Ausgabe Verzeichnis generiert werden. Typische Namen für Ausgabe-Verzeichnisse sind: • output - wegen Ausgabe • class - wegen Class-Dateien • bin - wegen binärem Code (Byte-Code) Aber auch andere Namen, die Assoziationen zu Ausgabe, Byte-Code oder Class-Dateien erzeugen, sind okay. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 98 / 409 Ansonsten gilt nur: das in Kapitel 4.3.3 erklärte anwenden - und fertig ist die Aufgabe. 5 Elementare Datentypen und Variablen 5.1 Datentypen 5.1.1 Datentypen Java ist eine statisch typisierte Sprache, d.h. jedes Objekt, Variable, Konstante, Literal, usw. hat einen eindeutigen Typ, der schon zur Compilezeit feststeht. Java unterscheidet zwischen elementaren und benutzerdefinierten Datentypen. • Elementare Datentypen sind fest in die Sprache eingebaut – siehe Kapitel 5.2. • Benutzerdefinierte Datentypen sind Klassen, Enums bzw. Interfaces – siehe Kapitel 11, Kapitel 12.6 und Kapitel 14.13. Java unterscheidet zwischen statischen und dynamischen Typen: • Der statische Typ ist der Typ, den der Compiler sieht. • Der dynamische Typ ist der wahre Typ des Objekts zur Laufzeit. Einen Unterschied zwischen statischen und dynamischen Typ kann es bei Objekten im Kontext von Vererbung geben. Diese wird uns ab Kapitel 14 begegnen. 5.1.2 Typ-Umwandlungen In Java können sogenannte Typ-Umwandlungen (auch Casts bzw. Type-Casts genannt) vorgenommen werden. Dies ist notwendig wenn ein Quelltyp nicht zum Zieltyp passt, z.B. bei einer Zuweisung oder einem Funktionsaufruf. Eine Typ-Umwandlung wird durch die Angabe des Zieltyps in Klammern vor dem Quellausdruck vorgenommen. byte b = (byte) 'A'; Hinweise: • Sie sollten sich bemühen, Code zu schreiben, der keine oder nur wenige TypUmwandlungen benötigt. • In Java sind nur Typ-Umwandlungen erlaubt, die syntaktisch Sinn machen. • Typ-Umwandlungen gibt es nur im Kontext der elementaren Datentypen und innerhalb von Vererbungs-Hierachien – siehe Kapitel 14.10. • Typ-Umwandlungen zwischen elementaren Datentypen werden benötigt, wenn der Quellausdruck nicht ohne Datenverlust in den Zieltyp paßt – dann müssen Sie diese Wandlung explizit anfordern. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 99 / 409 • Typ-Umwandlungen innerhalb von Vererbungs-Hierachien können zur Laufzeit schief gehen, sie werfen dann eine Exception – siehe Kapitel 14.10. 5.2 Elementare Datentypen Folgende elementaren Datentypen sind in Java enthalten: Name void char boolean byte short int long float double Grösse / Byte 2 1 Bit 1 2 4 8 4 8 Default \u0000 false 0 0 0 0 0.0 0.0 Minimum \u0000 -128 -32768 -2147483648 -9223372036854775808 ± 1.40239846 E - 45 ± 4.94065645841246544 E - 324 Maximum \uFFFF 127 32767 2147483647 9223372036854775807 ± 3.40282347 E + 38 ± 1.79769313486231570 E + 308 5.2.1 void void ist ein Dummy-Typ, um bei Funktionen anzuzeigen, dass sie nichts zurückgeben. void fct() { } // Diese Funktion gibt nichts zurueck 5.2.2 char Ein char enthält ein Zeichen im Unicode Zeichensatz – d.h. 2 Byte Grösse. • Zeichenkonstanten werden in einfache Hochkommata gesetzt. • Als Zeichenkonsten sind normalen Zeichen, Escape-Sequenzen (z. B. ‘\n‘ für einen Zeilenumbruch oder ‘\t’ für einen Tabulator) und Unicode Sequenzen (die Angabe muss hexa-dezimal erfolgen, z. B. '\u03a3') erlaubt. char c1 = 'A'; char c2 = '\n'; char c3 = '\u03a3'; // Zeichen A // Zeilenumbruch // Griechisches Summenzeichen (dezimal 931) Ein char kann ohne Typ-Cast einem int, long, float und double zugewiesen werden. Ein char kann mit explizitem Typ-Cast einem byte und short zugewiesen werden – denken Sie aber an den möglichen Datenverlust! byte b = (byte) 'A'; long l = 'A'; Umgekehrt kann einem char nur mit Typ-Cast ein integraler- oder Fliesskomma-Wert zugewiesen werden – denken Sie aber an den möglichen Datenverlust! char c1 = (char) 42; char c2 = (char) 3.14; © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 100 / 409 Achtung – vielleicht haben Sie sich gewundert, dass ein char und ein short nicht zuweisungskompatibel sind, sondern ein Typ-Cast nötig ist, wo doch beide Typen ein Grösse von 2 Byte haben. Aber bedenken Sie, dass ein char den Wertebereich von ‘\u0000’ bis ‘\uFFFF’ umfasst, d.h. keine negativen Zahlen aufweist, während ein short den Wertebereich von ‘-32768’ bis ‘+32767’ umfasst. 5.2.3 boolean Der Typ „boolean“ ist der Typ für Wahrheitswerte, daher wenn ein Zustand nur falsch („false“) oder richtig (true) sein kann. Wir benutzen ihn auch häufig als Kriterium in den Java Konstrollstrukturen – wie z.B. If-Anweisungen oder Schleifen, siehe Kapitel 7. Ein boolean Wert kann nur „true“ oder „false“ enthalten28. boolean flag = true; boolean okay = false; Ein boolean Wert kann keiner Variablen eines anderen Typs zugewiesen werden, auch nicht durch Typ-Cast’s. int i1 = true; int i2 = (int)true; // Compiler-Error // Compiler-Error Hinweis – diese Idee der Zuweisung eines Boolean an einen Int, wie auch die Idee im folgenden Beispiel, mag Sie vielleicht etwas verwundern. Wenn ja, dann ist das gut – und dann vergessen Sie dies direkt wieder. Aber es gibt Programmier-Sprachen, bei denen so etwas möglich ist – und für all die, die solche Sprachen kennen: in Java geht dies nicht. Auch umgekehrt kann kein Wert eines anderen Typs einer boolean Variable zugewiesen werden, oder als boolean Wert benutzt werden, auch nicht durch Typ-Cast’s. boolean b = (boolean)1; int i=0; if (i) { // Compiler-Error // Compiler-Error 5.2.4 Integrale Typen Als integrale Typen stehen byte, short, int und long zur Verfügung. Zahlen-Literale sind normalerweise immer int, können aber auch einem byte oder short zugewiesen werden, wenn sie den entsprechenden Zahlenbereich nicht überschreiten. long-Zahlen-Literale werden durch ein ‘l‘ oder ‘L‘ hinter der Zahl deklariert. byte b = 13; short s = 42; int i = 1234; long l = 123456789012345L; 28 true und false sind Schlüsselwörter © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 101 / 409 In Richtung der größeren Typen kann problemlos eine Zuweisung statt finden. In Richtung der kleineren nur über einen expliziten Typ-Cast - denken sie aber an den möglichen Datenverlust! int i = 17; short s = (short)i; long l = i; Jeder integrale Typ kann problemlos jedem Fliesskomma Typ zugewiesen werden – sie können dabei aber Genauigkeit verlieren! long l = 123456789012345L float f = l; double d = l; 5.2.5 Fliesskomma Typen Als Fliesskomma Typen stehen float und double zur Verfügung. Fliesskomma-Literale unterscheiden sich von den integralen Zahlen-Literalen dadurch, dass sie einen Punkt enthalten. Defaultmässig ist ein Fliesskomma-Literal immer vom Typ double, durch ein nachgestelltes ‘f‘ oder ‘F‘ wird es zu einem float Literal. float f = 3.14F; double d = 3.14; Ein float-Wert kann problemlos einem double zugewiesen werden, umgekehrt geht dies nur mit einem expliziten Typ-Cast - denken sie aber an den möglichen Datenverlust! float f1 = 3.14F; double d1 = 3.14; float f2 = (float)d1; double d2 = f1; 5.3 Variablen Variablen-Definiton Syntax: [Modifizierer] <Typ> <Name> [ = <Initialisierer>] ; Bsp: public final int i; private String name = "Albert Einstein"; double d = 3.1415; String s; Variablen können nur in Klassen und in Funktionen definiert werden. • Variablen in Klassen sind entweder Felder (Attribute oder Klassen-Variablen) – siehe Kapitel 11.4.6 und 12.5. • Variablen in Funktionen sind lokale Variablen – siehe Kapitel 8.3. Variablen werden einfach über ihren Namen angesprochen, und können z.B. ausgewertet, verglichen und neu gesetzt werden. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 102 / 409 public class Beispiel { public static void main(String[] args) { String name = "Albert Einstein"; System.out.println("Mein Name ist " + name); } } Mehr über Variablen findet sich in den Kapiteln 5.4, 8.3, 8.4, 11.4.6 und 12.5. 5.4 Wert- und Referenz-Semantik Variablen unterliegen je nach Typ einer unterschiedlichen Semantik. • Variablen eines elementaren Datentyps (siehe Kapitel 5.2) unterliegen einer sogenannten Wert-Semantik. • Variablen eines beliebigen anderen Datentyps unterliegen der Referenz-Semantik. 5.4.1 Wert-Semantik Eine Variable hat Wert-Semantik, wenn sie den zu speichernden Wert direkt enthält, und alle Aktionen auf der Variablen direkt auf dem Wert statt finden. Nehmen wir z.B. eine Int-Variable „v1“, die mit dem Wert „42“ initialisiert wird. int v1 = 42; Für diese Variable wird irgendwo im Speicher ein 4 Byte großer Bereich reserviert, der den Wert „42“ enthält, und der über den Namen „v1“ angesprochen werden kann. Wird jetzt eine zweite Int-Variable „v2“ definiert, die mit der Variablen „v1“ initialisiert wird, so bekommt „v2“ eine Kopie des Wertes von „v1“. Jede Variable enthält ihren eigenen Wert. int v1 = 42; int v2 = v1; System.out.println(v1); System.out.println(v2); // 42 // 42 Das jede Variable ihren eigenen Wert enthält wird dann deutlich, wenn wir den Wert einer Variablen ändern – die andere behält ihren alten Wert bei. int v1 = 42; int v2 = v1; System.out.println(v1); System.out.println(v2); // 42 // 42 v1 = 23; System.out.println(v1); System.out.println(v2); © Detlef Wilkening 1997-2016 // 23 <- veraendert // 42 <- nicht veraendert http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 1. Seite 103 / 409 2. v1: 42 3. v1: 42 v1: 23 v2: 42 v2: 42 Abb. 5-1 : Wert-Semantik Werte-Semantik liegt aber nicht nur bei einer Initialisierung oder einer Zuweisung vor, sondern z.B. auch bei Vergleichen (siehe auch Kapitel 6.2) und Funktions-Aufrufen (siehe Kapitel 8.4). Hier ein Beispiel mit zwei Vergleichen. int v1 = 42; int v2 = 42; if (v1==v2) { System.out.println("v1 und v2 sind gleich"); } if (v1==42) { System.out.println("v1 ist gleich 42"); } Ausgabe v1 und v2 sind gleich v1 ist gleich 42 In beiden Fällen wird der Wert von „v1“ (d.h. die „42“) mit dem Wert des zweiten Operanden verglichen. Diese Wert-Semantik gilt für alle Variablen eines elementaren Datentyps (d.h. char, boolean, byte, short, int, long, float oder double), unabhängig davon ob die Variable eine lokale Variable, ein Parameter, eine Objekt- oder Klassen-Variable ist. 5.4.2 Referenz-Semantik Abgesehen von den elementaren Datentypen werden alle Objekte (auch Arrays) in Java mit einer Referenz-Semantik angesprochen: • Eine Variable auf einen nicht-elementaren Datentyp enthält nicht das Objekt selber, sondern nur eine Referenz auf das Objekt, man spricht auch von Referenz-Variablen. • Eine Referenz-Variable wird mit null initialisiert29. • Vor der Nutzung einer Referenz-Variablen muss ihr ein Objekt zugewiesen werden. • ‚Normale‘ Referenz-Variablen-Zuweisungen (Operator =) kopieren das Objekt nicht, sondern erzeugen nur eine weitere Referenz auf das Objekt. 29 null ist in Java ein Schlüsselwort und ist der Defaultwert von Referenz-Variablen. Er hat nichts mit dem Wert ‘0‘ der elementaren Datentypen zu tun. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 • • • Seite 104 / 409 Bei Funktionsaufrufen unterliegen natürlich auch die Parameter und der Rückgabewert von nicht-elementaren Datentypen der Referenz-Semantik. Z. B. können also die OriginalObjekte von der aufgerufenen Funktion problemlos verändert werden. Der Zugriff auf Attribute und Funktionen von Objekten erfolgt über die Referenz-Variable und den Punkt-Operator. Der Vergleich zweier Referenz-Variablen vergleicht nicht die Werte der referenzierten Objekte (Wertgleichheit, bzw. tiefer Vergleich), sondern nur die Gleichheit des referenzierten Objekts (d.h. Identität, bzw. flacher Vergleich). Achtung – bei Strings kann es hier ein unerwartetes Verhalten geben – siehe unten bzw. Kapitel 6.2. StringBuffer buffer1 = null; if (buffer1==null) System.out.println("buffer1 ist noch null"); else System.out.println("buffer1 ist: " + buffer1); // (1) buffer1 = new StringBuffer("Hallo"); if (buffer1==null) System.out.println("buffer1 ist noch null"); else System.out.println("buffer1 ist: " + buffer1); // (2) StringBuffer buffer2 = buffer1; System.out.println("buffer2 ist: " + buffer2); // (3) buffer1.append(" Welt"); System.out.println("buffer1 ist: " + buffer1); System.out.println("buffer2 ist: " + buffer2); // (4) Ausgabe buffer1 buffer1 buffer2 buffer1 buffer2 ist noch null ist: Hallo ist: Hallo ist: Hallo Welt ist: Hallo Welt 1. buffer1: null 2. StringBuffer buffer1: => 3. "Hallo" StringBuffer buffer1: => buffer2: => 4. "Hallo" StringBuffer buffer1: => buffer2: => "Hallo Welt" Abb. 5-2 : Referenz-Semantik © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 105 / 409 Achtung – auch Strings sind Objekte und daher führt der Vergleichs-Operator „==“ nur einen Identitätsvergleich durch, und keinen Wert-Vergleich. Dies ist ein gern gemachter Fehler, der um so kritischer ist, da der Code manchmal sogar funktioniert – siehe Kapitel 6.2. Wird versucht über eine Referenz-Variable mit dem Wert „null“ auf das referenzierte Objekt zuzugreifen (das ja nicht da ist, da ja nichts (null) referenziert wird), so wird dieser Fehler javatypisch mit einer Exception („java.lang.NullPointerException“) gemeldet. String s = null; s.length(); // Wirft NullPointerException 6 Operatoren Folgende Operatoren stellt Java zur Verfügung. Vorr. 1 Operator . [] (args) ++, -- 2 ++, -+, - Operandentyp(en) Objekt Array, Integer Funktion arithmetisch As. L L L L Operation Zugriff auf Objekt-Element Zugriff auf Array-Element Funktionsaufruf Postinkrement, -dekrement (unär) ~ arithmetisch arithmetisch integral R R R Präinkrement, -dekrement (unär) unäres Plus, unäres Minus bitweises Komplement (unär) ! boolean R logisches Komplement (unär) (logisches Not) 3 new (Typ) Klasse beliebig R R Objekterzeugung Typumwandlung 4 *, /, % arithmetisch L Multiplikation, Division, Rest 5 +, + arithmetisch String L L Addition, Subtraktion String-Verkettung 6 << >> >>> integral integral integral L L L Links-Shift Rechts-Shift mit Vorzeichen-Erweiterung Rechts-Shift mit Null-Erweiterung 7 <, <= >, >= instanceof arithmetisch arithmetisch Typ L L L kleiner, kleiner-gleich grösser, grösser-gleich Typvergleich 8 == != == != einfach einfach Objekt Objekt L L L L gleich (identische Werte) ungleich (unterschiedliche Werte) gleich (identisches Objekt) ungleich (unterschiedliche Objekte) 9 & & integral boolean L L bitweises AND boolsches AND (alle Operanden werden ausgewertet) 10 ^ integral L bitweises XOR © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 106 / 409 ^ boolean L boolsches XOR (alle Operanden werden ausgewertet) 11 | | integral boolean L L bitweises OR boolsches OR (alle Operanden werden ausgewertet) 12 && boolean L bedingtes AND (Abbruch bei feststehendem Ergebnis) 13 || boolean L bedingtes OR (Abbruch bei feststehendem Ergebnis) 14 ?: boolean, beliebig R bedingter (ternärer) Operator R R Zuweisung Zuweisung mit Operation 15 = beliebig *=, /=, %=, beliebig +=, -=, <<=, >>=, >>>=, &=, ^=, |= • Die meisten dieser Operatoren sollten intuitiv klar sein, da Sie Programmier-Erfahrung mitbringen. • Ein paar Operatoren sind nicht zwingend intuitiv in ihrer Benutzung und Semantik, so dass sie im Weiteren besprochen werden. • Ein weiterer Teil der Operatoren macht erst im späteren Verlauf der Vorlesung Sinn, da uns hier noch das entsprechende notwendige Sprachverständnis fehlt. Sie werden in den jeweiligen Themen-Kapiteln vorgestellt, z.B. der Operator „instanceof“ in Kapitel 14.10. 6.1 Zuweisungs-Operatoren Der normale Zuweisungs-Operator ‘=’ weist den Wert des rechten Ausdrucks der Variablen auf der linken Seite zu. Zusätzlich gibt es in Java noch Zuweisungs-Operatoren mit Operationen. Die Anweisung ‘erg #= arg;’ entspricht dabei immer der Anweisung ‘erg = erg # (arg);’30. int i = 7; int j = 4; i += 3*j; System.out.println(i); // entspricht: // Ausgabe: 19 i = i + (3*j); 6.2 Gleich- und Ungleich-Operatoren Bei den elementaren Datentypen vergleichen die Operatoren ‘==’ und ‘!=’ den Wert, während bei Objekten die Identität (gleiches bzw. verschiedenes Objekt) verglichen wird – siehe auch z.B. Kapitel 5.4. Fangen wir mit einem Beispiel an, wo wir sehen, dass bei int’s mit den Werten gearbeitet wird: int i = 7; int j = 7; Durch die angegebene implizite Klammerung um ‘arg’ gibt es niemals Probleme mit den Operator-Prioritäten - es wird immer Gesamt-‘arg’ auf ‘erg’ angewandt. 30 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 if (i==j) System.out.println("Gleich"); else System.out.println("Ungleich"); j = 8; if (i==j) System.out.println("Gleich"); else System.out.println("Ungleich"); Seite 107 / 409 // Ausgabe: Gleich // Ausgabe: Ungleich Ganz anders ist das Verhalten bei Objekten, wo mit „==“ bzw. „!=“ die Identität verglichen wird: StringBuilder sb1 = new StringBuilder("sb"); StringBuilder sb2 = new StringBuilder("sb"); if (sb1==sb2) System.out.println("Gleich"); else System.out.println("Ungleich"); // Ausgabe: Ungleich sb2 = new StringBuilder("xxx"); if (sb1==sb2) System.out.println("Gleich"); else System.out.println("Ungleich"); // Ausgabe: Ungleich sb2 = sb1; if (sb1==sb2) System.out.println("Gleich"); else System.out.println("Ungleich"); // Ausgabe: Gleich Achtung – dies ist ein gern gemachter Fehler. Wert-Vergleiche auf Objekten müssen meist mit der Element-Funktion „equals“ gemacht werden, da „==“ nur die Identität vergleicht. Leider muß man hier sehr vorsichtig sein, da es Klassen gibt bei denen auch „equals“ nur die Identität vergleicht und nicht den Wert. StringBuilder sb1 = new StringBuilder("sb"); StringBuilder sb2 = new StringBuilder("sb"); if (sb1.equals(sb2)) System.out.println("Gleich"); else System.out.println("Ungleich"); sb2 = new StringBuilder("xxx"); if (sb1.equals(sb2)) System.out.println("Gleich"); else System.out.println("Ungleich"); sb2 = sb1; if (sb1.equals(sb2)) System.out.println("Gleich"); else System.out.println("Ungleich"); // Ausgabe: Gleich // Ausgabe: Ungleich // Ausgabe: Gleich Achtung – ganz gefährlich wird es bei Strings. In Kapitel 9.1 werden wir lernen, dass Strings „immutable“, d.h. unveränderlich, sind. Daher kann die JVM gleiche Strings ohne Gefahr zusammenlegen, um z.B. Speicher zu sparen. Daher können einzelne – eigentlich nicht identische Objekte – identisch sein. Es kann also sein, dass Sie in Ihrem Programm bei Strings fehlerhafterweise „==“ statt „equals“ verwenden, und es erstmal läuft. Gefährlich ist dies © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 108 / 409 deshalb, weil dieses Verhalten nicht zugesichert ist, sondern sich in anderen JVMs wieder ändern kann. String s1 = "Hallo"; String s2 = "Hallo"; // Kann vielleicht auf das gleiche Objekt zeigen if (s1==s2) { System.out.println("Identisch "); } else { System.out.println("NICHT identisch "); } // Identitaets-Abfrage if (s1.equals(s2)) { System.out.println("Gleicher Wert"); } // Defenitiv Wert-Vergleich mögliche Ausgabe NICHT identisch Gleicher Wert 6.3 Geteilt- und Modulo-Operator Während die normalen mathematischen Operatoren (plus, minus und mal) in ihrem Verhalten ziemlich intuitiv sind, ist dies beim Geteilt- und Modulo-Operator vielleicht nicht so. 6.3.1 Geteilt-Operator Im Prinzip arbeitet auch der Geteilt-Operator ganz intuitiv, außer man wendet ihn auf integrale Typen (byte, short, int und long) an und erwartet dann ein Fließkomma-Ergebnis (z.B. double). Das funktioniert so nicht – die Division bleibt im Bereich der Typen der Operanden, und das betrifft auch das Ergebnis. Teilen Sie Integer durch Integer, so ist das Ergebnis wieder ein Integer – selbst wenn das Ergebnis damit nicht exakt ist. Das Ergebnis ist nur der ganz-teilige Anteil der Division, den Rest kann man mit dem Modulo-Operator bestimmen – siehe Kapitel 6.3.2. System.out.println("10 System.out.println("10 System.out.println("10 System.out.println("10 System.out.println("10 System.out.println("10 System.out.println("10 System.out.println("10 System.out.println("10 System.out.println("10 System.out.println("10 System.out.println("10 Ausgabe 10 / 1 10 / 2 10 / 3 10 / 4 10 / 5 10 / 6 10 / 7 10 / 8 10 / 9 -> -> -> -> -> -> -> -> -> / 1 -> " / 2 -> " / 3 -> " / 4 -> " / 5 -> " / 6 -> " / 7 -> " / 8 -> " / 9 -> " / 10 -> " / 11 -> " / 12 -> " + + + + + + + + + + + + (10 (10 (10 (10 (10 (10 (10 (10 (10 (10 (10 (10 / / / / / / / / / / / / 1)); 2)); 3)); 4)); 5)); 6)); 7)); 8)); 9)); 10)); 11)); 12)); // // // // // // // // // // // // -> 10 -> 5 -> 3 -> 2 -> 2 -> 1 -> 1 -> 1 -> 1 -> 1 -> 0 -> 0 10 5 3 2 2 1 1 1 1 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 109 / 409 10 / 10 -> 1 10 / 11 -> 0 10 / 12 -> 0 Wie dividiert aber nun zwei Integer-Zahlen so, daß man das korrekte Fließkomma-Zahlen Ergebnis erhält? Ganz einfach: man muß vor der Division mindestens einen der beiden Operanden in den gewünschten Fließkomma-Zahlen-Typ („float“ oder „double“) konvertieren. final int divisor = 7; final int divident = 2; int res1 = divisor / divident; double res2 = (double) divisor / divident; double res3 = divisor / (double) divident; double res4 = (double) divisor / (double) divident; System.out.println("res1 System.out.println("res2 System.out.println("res3 System.out.println("res4 Ausgabe res1 -> res2 -> res3 -> res4 -> -> -> -> -> " " " " + + + + res1); res2); res3); res4); // // // // -> -> -> -> 3 3,5 3,5 3,5 3 3.5 3.5 3.5 Hinweis – für Konvertierungen zwischen elementaren Datentypen siehe Kapitel 5.1.2. 6.3.2 Modulo-Operator Der Modulo-Operator kann nur auf integrale Typen angewendet werden, und liefert dann den Rest der Integer-Division – siehe Kapitel 6.3.1. System.out.println("10 System.out.println("10 System.out.println("10 System.out.println("10 System.out.println("10 System.out.println("10 System.out.println("10 System.out.println("10 System.out.println("10 System.out.println("10 System.out.println("10 System.out.println("10 Ausgabe 10 % 1 10 % 2 10 % 3 10 % 4 10 % 5 10 % 6 10 % 7 10 % 8 10 % 9 10 % 10 10 % 11 10 % 12 -> -> -> -> -> -> -> -> -> -> -> -> % 1 -> " % 2 -> " % 3 -> " % 4 -> " % 5 -> " % 6 -> " % 7 -> " % 8 -> " % 9 -> " % 10 -> " % 11 -> " % 12 -> " + + + + + + + + + + + + (10 (10 (10 (10 (10 (10 (10 (10 (10 (10 (10 (10 % % % % % % % % % % % % 1)); 2)); 3)); 4)); 5)); 6)); 7)); 8)); 9)); 10)); 11)); 12)); // // // // // // // // // // // // -> 0 -> 0 -> 1 -> 2 -> 0 -> 4 -> 3 -> 2 -> 1 -> 0 -> 10 -> 10 0 0 1 2 0 4 3 2 1 0 10 10 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 110 / 409 6.4 Bitweises AND, OR, XOR Überarbeiten todo Die Operatoren ‘&’, ‘|’ und ‘^’ arbeiten sowohl mit integralen als auch mit boolschen Datentypen zusammen. Bei integralen Datentypen werden die interne Darstellung bitweise miteinander verknüpft. AND Op. 1 0 0 1 1 Op.2 0 1 0 1 OR Erg. 0 0 0 1 Op. 1 0 0 1 1 Op.2 0 1 0 1 XOR Erg. 0 1 1 1 Op. 1 0 0 1 1 Op.2 0 1 0 1 Erg. 0 1 1 0 6.5 Boolsches bzw. bedingtes AND, OR, XOR Bei boolschen Operanden werden die Operanden logisch miteinander verknüpft, wobei immer alle Operanden ausgewertet werden. Bei den bedingten Operatoren ‘&&’ und ‘||’ werden die Operanden auch logisch miteinander verknüpft, aber die Auswertung wird abgebrochen wird, sobald das Ergebnis feststeht. Es kann daher passieren, dass nicht alle Operanden ausgewertet werden. public static boolean fct(int i) { System.out.println("fct(" + i + ')'); return true; } public static void main(String[] args) { boolean erg, b = true; erg = b & fct(1); erg = b | fct(2); erg = b ^ fct(3); erg = b && fct(4); erg = b || fct(5); b = false; erg = b & fct(6); erg = b | fct(7); erg = b ^ fct(8); erg = b && fct(9); erg = b || fct(10); } Ausgabe fct(1) fct(2) fct(3) fct(4) fct(6) fct(7) fct(8) fct(10) © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 111 / 409 6.6 Prä- und Post-Inkrement und -Dekrement Operatoren Den Operator ‘++’ gibt es in Prä- und Postfix-Notation. Er ist für alle arithmetischen Typen definiert und entspricht einer Erhöhung um ‘1’. Analog entspricht der ‘—‘ Operator einer Erniedrigung um ‘1’. int i = 7; i++; System.out.println(i); ++i; System.out.println(i); i--; System.out.println(i); --i; System.out.println(i); // Ausgabe: 8 // Ausgabe: 9 // Ausgabe: 8 // Ausgabe: 7 Im obigen Beispiel ist die Unterscheidung in Prä- und Postfix-Notation egal, aber in anderen Zusammenhängen ist sie es nicht. Diese Operatoren können direkt in arithmetischen Ausdrücken benutzt werden - in so einem Fall entscheidet die Notation, ob die Variable zuerst ausgewertet und dann verändert wird, oder umgekehrt: • Präfix-Notation ‘++x’ bzw. ‘–-x’ 1. verändern von ‘x’ 2. auswerten von ‘x’ • Postfix-Notation ‘x++’ bzw. ‘x--’ 1. auswerten von ‘x’ 2. verändern von ‘x’ int i = 7; int j = ++i; System.out.println(i + " " + j); j = i--; System.out.println(i + " " + j); // Ausgabe: 8 8 // Ausgabe: 7 8 Im Gegensatz zu einigen anderen Programmiersprachen wie z.B. C und C++ ist auch bei Verwendung mehrere dieser Operatoren in einer Anweisung das Ergebnis exakt definiert, da es in Java eine eindeutige Abarbeitung von Ausdrücken gibt: zuerst werden bei einer Anweisung die Operanden von links nach rechts ausgewertet, dann wird der Rest der Anweisung durchgeführt – mit der Ausnahme von Operanden bei bedingtem Und und Oder! int i = 2; int j = ++i + ++i * ++i; System.out.println(i); // entspricht 3+4*5 // Ausgabe: 23 6.7 Schiebe-Operatoren Die Schiebe-Operatoren ‘<<’, ‘>>’ und ‘>>>’ wirken nur auf integrale Datentypen. Der Operator ‘<<’ schiebt die Bits des integralen Operanden um n-Bits nach links, d.h. er bewirkt bei allen Zahlen (negativ und positiv) eine Multiplikation mit 2. int i = 3; © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 int j = -5; System.out.println((i<<1) + " " + (j<<2)); Seite 112 / 409 // Ausgabe: 6 -20 Der Operator ‘>>’ schiebt die Zahl um n-Bits nach rechts mit Vorzeichen-Erweiterung. D.h. das Vorzeichen der Zahl bleibt erhalten. Dagegen füllt der Operator ‘>>>’ die Zahl immer mit NullBits auf - er betrachtet die Zahl quasi immer als vorzeichenlos. int i = 5; int j = -5; int k = -5; System.out.println((i>>1)+" "+(j>>1)+" "+(k>>>1)); // Ausgabe: 2 -3 2147483645 6.8 Frage-Zeichen Operator todo 6.9 Aufgaben 6.9.1 Aufgabe „Inkrement-Operator“ Schreiben Sie ein kleines Programm, das den Unterschied im Verhalten der Prä- und PostInkrement Operatoren („++“) aus Kapitel 6.6 verdeutlicht, d.h. wo ein Vertauschen der Operatoren zu einem anderen Ergebnis führt. Lösung siehe Kapitel 6.10. 6.10 Lsg. zu Aufgabe „Inkrement-Operator“ – Kap. 6.9.1 Um das unterschiedliche Verhalten der Operatoren für Prä- und Post-Inkrement zu sehen, ist es wichtig zu verstehen, dass die Addition um „1“ nicht die einzige Aktion der Anweisung ist. Die beiden folgenden Zeilen unterscheiden sich semantisch (d.h. in ihrer Wirkung) nicht, da hier die Addition die einzige veränderne Aktion in der Anweisung ist. ++i; i++; // Diese beiden Zeilen haben so fuer sich die gleiche Wirkung, // d.h. hat ein Vertauschen der Operatoren hat hier keine Auswirkung Um den Unterschied zu sehen, muss man den Wert des „++“ Ausdrucks in der Anweisung auswerten. Dazu führen wir eine zweite Variable ein, der wir den „++“ Ausdruck zuweisen. j = ++i; j = i++; Jetzt müssen wir nur noch für gleiche Startwerte sorgen, und das Ergebnis ausgeben. Hier am Beispiel des Prä-Inkrement-Operators: int i = 4; int j = 0; System.out.println("Prae-Inkrement"); j = ++i; System.out.println("j = ++i"); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 113 / 409 System.out.println("=> i: 4 -> " + i); System.out.println("=> j: 0 -> " + j); Jetzt das gleiche noch für den Post-Inkrement-Operator, und dann das ganze in eine Klasse und eine Main-Funktion – schon ist das Programm fertig: public class Appl { public static void main(String[] args) { int i = 4; int j = 0; System.out.println("Prae-Inkrement"); j = ++i; System.out.println("j = ++i"); System.out.println("=> i: 4 -> " + i); System.out.println("=> j: 0 -> " + j); i = 4; j = 0; System.out.println("\nPost-Inkrement"); j = i++; System.out.println("j = i++"); System.out.println("=> i: 4 -> " + i); System.out.println("=> j: 0 -> " + j); } } Ausgabe Prae-Inkrement j = ++i => i: 4 -> 5 => j: 0 -> 5 Post-Inkrement j = i++ => i: 4 -> 5 => j: 0 -> 4 An der Ausgabe sieht man sehr schön den Unterschied: • Beim Prä-Inkrement wird zuerst das „i“ erhöht (von 4 auf 5), und dann der neue Wert von „i“ (5) der Variablen „j“ zugewiesen. Daher haben beide Variablen nach der Anweisung den Wert „5“. • Beim Post-Inkrement wird zuerst der bisherige Wert (4) des „i“ genommen und als Ergebnis des Ausdrucks benutzt, d.h. hier dem „j“ zugewiesen. „j“ hat also den Wert „4“. Erst danach wird „i“ erhöht, und bekommt damit den Wert „5“. Diese Erhöhung hat keinen Einfluß mehr auf den Rest dieser Anweisung. 7 Kontroll-Strukturen 7.1 Bedingter-Kontrollfluss – If Die wichtigste Kontroll-Struktur ist die If-Anweisung. Sie ermöglicht einen bedingten Sprung, d.h. in Abhängigkeit von einem Boolschen-Ausdruck unterschiedliche Pfade im Programm zu durchlaufen. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 114 / 409 7.1.1 Einfachste Form Die einfachste Form der If-Anweisung sieht folgendermaßen aus: Syntax: if (boolean-ausdruck) true-anweisung Sie beginnt mit dem Schlüsselwort „if“ – dann folgt in runden Klammern ein Ausdruck, der vom Typ „boolean“ sein muss. Abgeschlossen wird sie durch eine Anweisung. Diese Anweisung wird nur ausgeführt, wenn der Ausdruck zu „true“ ausgewertet wurde. Im Falle von „false“ wird die Anweisung übersprungen. int n = 5; if (n<7) System.out.println("n<7"); if (n>20) System.out.println("n>20"); if (n!=6) System.out.println("n!=6"); Ausgabe n<7 n!=6 7.1.2 Blöcke für mehrere Anweisungen Soll im If-Zweig mehr als eine Anweisung ausgeführt werden, so muß mit den geschweiften Klammern ein Block gebildet werden. Ein solcher Block steht dabei syntaktisch für eine einzelne Anweisung. int n = 5; if (n<7) { System.out.println("n<7"); System.out.println("n ist " + n); System.out.println("Und weiter geht es..."); } Ausgabe n<7 n ist 5 Und weiter geht es... Hinweis – dies gilt genauso für Schleifen oder andere Konstrukte, die normalerweise nur auf eine Anweisung wirken. 7.1.3 If-Else Anweisung Für den Fall, dass neben dem True-Fall auch beim False-Fall Aktionen ausgeführt werden sollen, kann auf die Anweisung (bzw. dem Block) des True-Falls das Schlüsselwort „else“ mit einer Anweisung für den False-Fall folgen. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 115 / 409 Syntax: if (boolean-ausdruck) true-anweisung else false-anweisung Beispiel: int n = 5; if (n<7) System.out.println("n<7"); else System.out.println("n>=7"); if (n>20) System.out.println("n>20"); else System.out.println("n<=20"); Ausgabe n<7 n<=20 Hinweis – auf für den False-Zweig gilt natürlich: Wenn mehr als eine Anweisung notwendig ist, dann muß ein Block genutzt werden – siehe Kapitel 7.1.2. 7.1.4 If-Anweisung ohne True-Zweig Es gibt keine Syntax, um eine If-Anweisung nur mit False-Zweig ohne True-Zweig zu implementieren. Ist dies gewünscht, so gibt es zwei Implementierungs-Möglichkeiten: • True-Zweig mit leerer Anweisung • Bool-Ausdruck in der If-Anweisung umformulieren True-Zweig mit leerer Anweisung In Java sind leere Anweisungen erlaubt – sie bestehen aus einem einzelnen Semikolon. Damit läßt sich der True-Zweig einer If-Anweisung sehr schnell abhandeln. int n = 10; if (n<7); // <= dieses Semikolen ist der leere True-Zweig else System.out.println("n>=7"); Ausgabe n>=7 Achtung – so ein leerer True-Zweig ist nicht besonders gut lesbar. Und so ein einzelnes Semikolon ist schnell überlesen bzw. verwirrend – von daher ist die Lösung nicht zu empfehlen. Formulieren Sie besser den Bool-Ausdruck um. Bool-Ausdruck in der If-Anweisung umformulieren Formulieren Sie den boolschen Ausdruck so um, dass er statt „true“ „false“ liefert und umgekehrt. Wenn sich der Ausdruck nicht umformulieren läßt, bzw. die Umformulierung © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 116 / 409 aufwändig und fehlerträchtig ist – dann nutzen Sie einfach den boolschen Not-Operator „!“ um den boolschen Ausdruck zu negieren. int n = 10; if (n>=7) System.out.println("n>=7"); // Ausdruck umformuliert if (!(n<7)) System.out.println("n>=7"); // Ausdruck negiert mit ! Ausgabe n>=7 n>=7 7.1.5 Mehrfach-Verzweigungen mit If-ElseIf Nicht immer gibt es nur einen True-Fall bzw. einen True- und einen False-Fall. Häufig hat man eine Art Auflistung von verschiedenen Fällen, d.h. eine Mehrfach-Verzweigung. In diesem Fällen nimmt man verschachtelte If-Anweisungen, die auch als If-ElseIf-Anweisung bezeichnet wird. Im Prinzip ist eine If-ElseIf-Anweisung nichts neues – hier wird nur der False-Zweig durch eine If-Anweisung dargestellt. Der Unterschied ist eine eigenständige Einrückungs-Konvention, bei der das „if“ direkt auf das „else“ folgt. Syntax: if (boolean-ausdruck) true-anweisung else if (boolean-ausdruck) true-2-anweisung else false-anweisung Beispiel: int age = 27; if (age<10) System.out.println("Kind im Alter von " + age); else if (age<18) System.out.println("Jugendliche(r) im Alter von " + age); else if (age<30) System.out.println("Junge(r) Erwachsene(r) im Alter von " + age); else if (age<70) System.out.println("Erwachsene(r) im Alter von " + age); else System.out.println("Alte(r) Frau/Mann im Alter von " + age); Ausgabe Junge(r) Erwachsene(r) im Alter von 27 Hinweis – auch hier ist der Else-Zweig natürlich optional. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 117 / 409 7.2 Mehrfach-Verzweigung – Switch Für spezielle Mehrfach-Verzweigungen in Abhängigkeit von einer integralen Variable gibt es in Java die Switch-Anweisung. Sie funktioniert für die Typen „char“, „byte“, „short“ und „int“, d.h. Typen mit integralen Werten mit max. 4 Byte Größe, und zusätzlich seit dem JDK 1.5 mit Enums (siehe Kapitel todo) und seit dem JDK 1.7 auch mit Strings. Außerdem kann der integrale Ausdruck nur auf Gleichheit zu Compile-Zeit Konstanten des entsprechenden Typs getestet werden. Wenn man mit diesen Einschränkungen leben kann, ist eine Switch-Anweisung einer If-Else-If Mehrfach-Verzweigung vorziehen. Syntax switch (ausdruck) { case constant1: anweisung; ... [break;] case constant2: anweisung; ... [break;] ... default: anweisung; ... [break;] } • ausdruck muss vom Typ „char“, „byte“, „short“, „int“, „Enum“ (seit JDK 1.5) oder „String“ (seit JDK 1.7) sein. • constant muss eine Compile-Zeit-Konstante vom entsprechenden Typ sein. public class Appl { public static void main(String[] args) { for (int i = 0; i < 5; i++) { switch (i) { case 0: System.out.println("Null"); break; case 1: System.out.println("Eins"); break; case 3: System.out.println("Drei"); break; default: System.out.println("Unbekannt"); } } } } Ausgabe Null Eins Unbekannt Drei Unbekannt © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 118 / 409 Erst die „break“ Anweisung beendet einen Case-Block. Ist kein „break“ vorhanden, so läuft der Programmfluss in den nächsten Case-Block hinein – auch in den Default-Block. Auch im Default-Block darf das „break“ stehen – hat hier aber keine Bedeutung – siehe Beispiel: for (int i=0; i<7; i++) { switch (i) { case 1: System.out.println("i ist break; case 2: case 3: System.out.println("i ist case 4: System.out.println("Fluss break; default: System.out.println("i ist break; } } 1"); 2 oder 3"); kommt aus 2/3 oder i ist 4"); weder 1,2,3 oder 4"); // Nicht notwendig Ausgabe i ist weder 1,2,3 oder 4 i ist 1 i ist 2 oder 3 Fluss kommt aus 2/3 oder i ist 4 i ist 2 oder 3 Fluss kommt aus 2/3 oder i ist 4 Fluss kommt aus 2/3 oder i ist 4 i ist weder 1,2,3 oder 4 i ist weder 1,2,3 oder 4 Seit JDK 1.5 kann man die Switch-Anweisung auch für Enums (siehe Kapitel 12.6) verwenden. Achtung, das Beispiel enthält mehrere Sprachmittel (u.a. die Enums), die erst in späteren Kapiteln eingeführt werden: public class Appl { private enum Color { RED, GREEN, BLUE }; private static void switchIt(Color c) { switch (c) { case RED: System.out.println("rot"); break; case GREEN: System.out.println("gruen"); break; case BLUE: System.out.println("blau"); break; } } public static void main(String[] args) { switchIt(Color.RED); switchIt(Color.GREEN); switchIt(Color.BLUE); } } Ausgabe rot gruen © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 119 / 409 blau Und seit JDK 1.7 funktioniert die Switch-Anweisungs auch mit Strings. public class Appl { private static void doStringSwitch(String s) { switch (s) { case "Text 1": System.out.println("Text 1 gefunden"); break; case "Text 2": System.out.println("Text 2 gefunden"); break; default: System.out.println("Unbekannter Text"); break; } } public static void main(String[] args) { doStringSwitch("Text 1"); doStringSwitch("Text 2"); doStringSwitch("Text 3"); } } Ausgabe Text 1 gefunden Text 2 gefunden Unbekannter Text Hinweis – der Java Compiler wandelt die Switch-Anweisungen mit Strings in eine Art „HashTable“ mit Sprunganweisung um. Daher wird die Switch-Anweisungen bei vielen Case-Fällen wahrscheinlich performanter sein als If-Else-If-Else-Anweisungen. 7.3 For-Schleife Syntax for (init; test; update) anweisung Beispiel for (int i=0; i<8; i++) { System.out.print(i); } Ausgabe 01234567 Hinweis – werden im Initialisierungs-Teil ein oder mehrere Variablen definiert, so sind diese nur im Scope der Schleife bekannt. for (int i=0; i<8; i++) { System.out.print(i); } System.out.print(i); © Detlef Wilkening 1997-2016 // Definition von i gilt nur fuer die Schleife // Compiler-Fehler – Variable i ist unbekannt http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 120 / 409 Hinweis – mit dem JDK 1.5 wurde ein weiterer For-Schleifentyp für Container und Arrays eingeführt – dieser wird im nächsten Kapitel 7.4 beschrieben. 7.4 For-Schleife für Container und Arrays In Java 5 (JDK 1.5) wurde ein neuer For-Schleifen-Typ für Container (siehe Kapitel 9.3) und Arrays (siehe Kapitel 10.3) eingeführt. Achtung – die folgenden beiden Programme können Sie hier noch nicht verstehen, da in ihnen sowohl Arrays als auch typisierte Container vorkommen. Da es aber hier im Kapitel u.a. um Schleifen geht, habe ich diesen Schleifen-Typ hier auch beschrieben – kommen Sie später nochmal zurück zu diesem Kapitel. Neuer Schleifentyp für Arrays: int[] a = new int[]{ 1, 2, 3, 5, 7, 11 }; for (int n : a) { System.out.print("" + n + ' '); } Ausgabe 1 2 3 5 7 11 Neuer Schleifentyp für Container: ArrayList<String> al = new ArrayList<String>(); al.add("eins"); al.add("zwei"); al.add("drei"); for (String s : al) { System.out.println("-> " + s); } Ausgabe -> eins -> zwei -> drei 7.5 While-Schleife Syntax while (boolan-ausdruck) anweisung Die While-Schleife wird solange wiederholt, wie der ausdruck true ist. Beispiel int i = 0; while (i<3) { System.out.print(i + " "); i++; } Ausgabe 0 1 2 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 121 / 409 7.6 Do-Schleife Syntax do anweisung while (boolean-ausdruck); Die Do-Schleife wird solange wiederholt, wie der Ausdruck true ist. Beispiel int i = 0; do { System.out.print(i + " "); i++; } while (i<3); Ausgabe 0 1 2 Im Gegensatz zur While-Schleife wird die Do-Schleife immer mindestens einmal durchlaufen. 7.7 Break- und Continue-Anweisung 7.7.1 Break Das Schlüsselwort „break“ in einer beliebigen Schleife sorgt für einen sofortigen Abbruch der Schleife. for (int i=0; i<20; i++) { System.out.print(i + " "); if (i>=5) break; } System.out.print("Schleifenende"); Ausgabe 0 1 2 3 4 5 Schleifenende Dies gilt nur für die innerste Schleife, falls Sie mehrere Schleifen ineinander verschachtelt haben. for (int i=0; i<2; i++) { for (int j=0; j<20; j++) { System.out.print(j + " "); if (j>=5) break; } System.out.print("Ende-von-j-Schleife\n"); } System.out.print("Schleifenende"); Ausgabe 0 1 2 3 4 5 Ende-von-j-Schleife 0 1 2 3 4 5 Ende-von-j-Schleife Schleifenende 7.7.2 Continue Das Schlüsselwort „continue“ in einer Schleife sorgt für einen sofortigen Sprung an den Schleifen-Kopf – wobei auch bei der Do-Schleife die Test-Bedingung zum Schleifen-Kopf zählt. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 122 / 409 Achtung – continue wirkt bei For-Schleifen anders als bei While- und Do-Schleifen, da sich bei diesen Schleifen die Wirkung des Schleifenkopfes unterscheidet: • Bei einer For-Schleife wird die Update-Ausdruck ausgeführt, und dann der Test-Ausdruck ausgewertet – mit einem möglichem Abbruch der Schleife. • Bei einer While- oder einer Do-Schleife wird nur der Test-Ausdruck ausgewertet – mit einem möglichem Abbruch der Schleife. System.out.println("For"); for (int i = 0; i < 7; i++) { System.out.print(i + " "); if (i == 3) { System.out.print(" ** "); i += 2; continue; } System.out.print(i + " - "); } System.out.println("\nWhile"); int i = 0; while (i < 7) { System.out.print(i + " "); if (i == 3) { System.out.print(" ** "); i += 2; continue; } System.out.print(i + " - "); i++; } System.out.println("\nDo"); i = 0; do { System.out.print(i + " "); if (i == 3) { System.out.print(" ** "); i += 2; continue; } System.out.print(i + " - "); i++; } while (i < 7); Ausgabe For 0 0 - 1 1 - 2 2 - 3 While 0 0 - 1 1 - 2 2 - 3 Do 0 0 - 1 1 - 2 2 - 3 ** 6 6 ** 5 5 - 6 6 ** 5 5 - 6 6 - Hinweis – auch „continue“ wirkt nur für die innerste Schleife. Wollen Sie zum Schleifenkopf einer äußeren Schleife springen, so müssen Sie eine gelabelte Sprung-Anweisung nutzen – siehe nächstes Kapitel. 7.8 Gelabelte Sprung-Anweisungen Gelabelte Sprung-Anweisungen ermöglichen ein break oder continue über mehrere SchleifenEbenen hinweg. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 123 / 409 Syntax break label; continue label; Dazu muß eine Schleife mit einem Label versehen werden – dies geschieht mit dem LabelNamen und einem folgenden Doppelpunkt. Wird jetzt innerhalb einer solchen Schleife eine gelabelte Sprunganweisung ausgeführt, so bezieht sie sich auf die Schleife mit dem passenden Label, auch wenn diese nicht die innerste ist. Beispiel outerLoop: for (int i=0; i<5; i++) { for (int j=0; j<3; j++) { System.out.print(i + " " + j + " / "); if (i+j > 3) break outerLoop; } } Ausgabe 0 0 / 0 1 / 0 2 / 1 0 / 1 1 / 1 2 / 2 0 / 2 1 / 2 2 / Hinweis – eine gelabelte Sprunganweisung darf natürlich nur innerhalb einer Schleife mit einem passenden Label stehen. Man kann mit ihnen also nicht in eine ganz andere Schleife springen. 7.9 Aufgaben 7.9.1 Aufgabe „Schleifen-Varianten“ Schreiben Sie ein Programm, dass dreimal hintereinander die Zahlen von 1 bis 5 ausgibt. Für die erste „1..5“ Ausgabe soll eine For-Schleife benutzt werden, für die zweite eine WhileSchleife, und für die dritte eine Do-Schleife. Jede „1..5“ Ausgabe soll in einer eigenen Zeile stehen, und die Zahlen sollen mit einem Leerzeichen untereinander getrennt sein. Ausgabe 1 2 3 4 5 1 2 3 4 5 1 2 3 4 5 Lösung siehe Kapitel 7.10. 7.9.2 Aufgabe „Teilbar?“ Schreiben Sie ein Programm, dass hintereinander die Zahlen von 1 bis 10 und jeweils eine Meldung, ob die Zahl durch 2 bzw. durch 3 teilbar ist, ausgibt. Benutzen Sie, um die Zahlen zu durchlaufen, eine For-Schleife. Mögliche Ausgabe: 1 nicht-durch-2-teilbar nicht-durch-3-teilbar 2 durch-2-teilbar nicht-durch-3-teilbar 3 nicht-durch-2-teilbar durch-3-teilbar 4 durch-2-teilbar nicht-durch-3-teilbar © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 5 nicht-durch-2-teilbar ... Seite 124 / 409 nicht-durch-3-teilbar Lösung siehe Kapitel 7.11. 7.9.3 Aufgabe „Lesbare Zahlen“ Schreiben Sie ein Programm, dass hintereinander die Zahlen von 1 bis 5 als lesbaren Text ausgibt. Benutzen Sie, um die Zahlen zu durchlaufen, eine For-Schleife. Jede Zahl-Ausgabe soll in einer eigenen Zeile stehen. Mögliche Ausgabe: eins zwei drei vier fuenf Vergeichen Sie zu dieser Aufgabe auch die Aufgaben 9.8.4 (Lösung in Kapitel 9.12) und 10.8.1 (Lösung in Kapitel 10.9). Lösung siehe Kapitel 7.12. 7.9.4 Aufgabe „Zahlen-Liste“ Schreiben Sie ein Programm, dass hintereinander die Zahlen von 1 bis 5 ausgibt. Die Zahlen sollen dabei durch einen Bindestrich ‚-‚ getrennt werden, und die gesamte Zahlenreihe soll von runden Klammern umgeben sein. Benutzen Sie, um die Zahlen zu durchlaufen, eine ForSchleife. Ausgabe: (1-2-3-4-5) Lösung siehe Kapitel 7.13. 7.9.5 Aufgabe „Mittelwert und Varianz“ Schreiben Sie ein Programm, das den Mittelwert und die Varianz von Gleitkomma-Zahlen berechnet. Der Benutzer gibt die Werte auf der Kommandozeile an. Der Mittelwert ist die Summe der Werte geteilt durch die Anzahl. Die Varianz ist die Summe der Quadrate der Werte geteilt durch die um eins reduzierte Anzahl. Fangen sie fehlerhafte Eingaben wieder sinnvoll ab. Denken Sie sich ein sinnvolles Abbruchkriterium aus, d.h. was muss der Benutzer eingeben, wenn er fertig mit der Eingabe ist. Um auch dieses Programm erstmal etwas einfacher zu gestalten, könnten Sie – ähnlich zu Aufgabe 4.8.5 – eine Version schreiben, die die Werte als Kommandozeilen-Argumente übergeben bekommt statt Sie einzulesen. Lösung siehe Kapitel 7.14. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 125 / 409 7.9.6 Aufgabe „Multiplikations-Matrix“ Schreiben Sie ein Programm, das zwei positive Integer-Zahlen einliest, und für diese (von bis inkl.) das Ein-mal-eins als Matrix ausgibt. Hinweise: • Fangen Sie fehlerhafte Eingaben ab. Diesen Hinweis gibt es hier jetzt zum letzten Mal, danach wird er als selbstverständlich vorausgesetzt. • Wie in Aufgabe 4.8.5 und Aufgabe 7.9.5 ist es für den Einstieg möglich, zuerst eine Version mit Kommandozeilen-Argumenten zu schreiben. Beispiel: Zahl1: 2 Zahl2: 4 | 2 3 4 ---+-----------2 | 4 6 8 3 | 6 9 12 4 | 8 12 16 Hinweis – für eine schönere Formatierung fehlt uns leider noch das Wissen. Lösung siehe Kapitel 7.15. 7.10 Lsg. zu Aufgabe „Schleifen-Varianten“ – Kap. 7.9.1 Diese Aufgabe ist relativ einfach, da sie nur einfach die drei Schleifen-Typen aus den Kapiteln 7.3, 7.5 und 7.6 implementieren müssen. Letzlich ist das ja auch der Sinn der Aufgabe, einmal mit den drei Schleifen-Typen von Java vertraut zu werden. Für einen Java Anfänger hat die Aufgabe vielleicht trotzdem zwei kleine Knackpunkte: • Die Ausgabe der Zahl müssen Sie mit „System.out.print“ machen (z.B. Zeile *), damit kein Zeilenumbruch erzeugt wird. Hinter jedem Schleifen-Typ muss dann ein Zeilenumbruch erzeugt werden – z.B. mit „System.out.println()“ (z.B. Zeile **). • Für alle drei Schleifen wird eine Zählvariable benötigt. Hier müssen Sie mit der Sichtbarkeit der Variablen (siehe u.a. Kapitel 5.3) aufpassen. Für die While- und die Do-Schleife benötigen Sie eine Zähl-Variable, die außerhalb der Schleife definiert ist (Zeile ***). Damit ist klar, dass Sie sie für die jeweilig zweite Schleife nicht mehr definieren müssen, sondern nur den Start-Wert neu setzen müssen (Zeile ****). Die For-Schleife ist hier eine Besonderheit. Da der Schleifenkopf zur Sichtbarkeit der Schleife gehört, können (und sollten Sie im Normalfall auch) Sie die Zähl-Variable dort definieren. Dann steht sie aber für die anderen beiden Schleifen-Typen nicht mehr zur Wiederverwendung zur Verfügung. Anders als das Beispiel könnten Sie hier also auch eine Zähl-Variable wie in Zeile (***) definieren, und sie auch für die For-Schleife © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 126 / 409 wiederverwenden. Etwas anderes passiert, wenn Sie die For-Schleife nicht als ersten, sondern als zweiten oder dritten Schleifen-Typ verwenden. In diesem Fall können Sie in der For-Schleife keine ZählVariable mehr definieren – zumindest nicht mit dem gleichen Namen – da Java keine mehrfachen Variablen-Namen innerhalb einer Funktion zuläßt (siehe Kapitel 8.3). public class Appl { public static void main(String[] args) { for (int i=1; i<6; i++) { System.out.print(i + " "); } System.out.println(); // (*****) // (*) // (**) int i = 1; while (i<6) { System.out.print(i++ + " "); } System.out.println(); // (***) i = 1; do { System.out.print(i++ + " "); } while (i<6); System.out.println(); // (****) } } Ausgabe 1 2 3 4 5 1 2 3 4 5 1 2 3 4 5 7.11 Lsg. zu Aufgabe „Teilbar?“ – Kap. 7.9.2 Die Aufgabe besteht aus zwei Teilen. • Zuerst müssen natürlich die Zahlen von 1 bis 10 durchlaufen werden. • Danach muss für jede Zahl geprüft werden, ob sie durch 2 bzw. 3 teilbar ist, und dann eine entsprechende Meldung ausgegeben werden. Teil 1 sollten sie mittlerweile problemlos implementieren können, da sie hierfür nur eine einfache Zählschleife, also eine For-Schleife benötigen. for (int i=1; i<=10; i++) { ... } Teil 2 setzt die richtige Idee voraus, wie man die Teilbarkeit testen kann. Die Lösung ist hier der Modulo-Operator „%“ – siehe Kapitel 6. Er gibt den Rest der Integer-Division zurück, und die ist „0“, wenn der erste Operand durch den zweiten Operanden teilbar ist. Mit einer entsprechenden Meldung sieht der Test auf Teilbarkeit durch 2 also z.B. so aus: if (i%2!=0) { System.out.print("nicht-"); } System.out.print("durch-2-teilbar"); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 127 / 409 Damit ist alles vorhanden – jetzt müssen die Teile nur noch zusammengefügt werden: • Klasse mit Main-Funktion • For-Schleife mit Ausgabe • Test auf Teilbarkeit auf „2“ und „3“ Hier das gesamte Programm: public class Appl { public static void main(String[] args) { for (int i=1; i<=10; i++) { System.out.print(i + " "); if (i%2!=0) { System.out.print("nicht-"); } System.out.print("durch-2-teilbar "); if (i%3!=0) { System.out.print("nicht-"); } System.out.println("durch-3-teilbar"); } } } Ausgabe: 1 nicht-durch-2-teilbar nicht-durch-3-teilbar 2 durch-2-teilbar nicht-durch-3-teilbar 3 nicht-durch-2-teilbar durch-3-teilbar 4 durch-2-teilbar nicht-durch-3-teilbar 5 nicht-durch-2-teilbar nicht-durch-3-teilbar 6 durch-2-teilbar durch-3-teilbar 7 nicht-durch-2-teilbar nicht-durch-3-teilbar 8 durch-2-teilbar nicht-durch-3-teilbar 9 nicht-durch-2-teilbar durch-3-teilbar 10 durch-2-teilbar nicht-durch-3-teilbar 7.12 Lsg. zu Aufgabe „Lesbare Zahlen“ – Kap. 7.9.3 Klar sollte sein: auch hier wird wieder eine Klasse mit Main-Funktion und einer For-Schleife benötigt. Bleibt nur die Frage offen: wie unterscheidet man auf die verschiedenen Zahlenwerte? Die Lösung ist ganz einfach: das Problem ist typisch für eine Switch-Anweisung (siehe Kapitel 7.2), die eine Unterscheidung für integrale Werte darstellt. Damit kann das Programm direkt hingeschrieben werden: public class Appl { public static void main(String[] args) { for (int i=1; i<=5; i++) { switch (i) { case 1: System.out.println("eins"); break; case 2: System.out.println("zwei"); break; case 3: System.out.println("drei"); break; © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 128 / 409 case 4: System.out.println("vier"); break; case 5: System.out.println("fuenf"); break; } } } } Hinweis – noch effizienter läßt sich das Programm mit Containern implementieren, die wir in Kapitel 9.3 kennen lernen werden. Dort wird sich das Problem auch noch mal als neue Aufgabe 9.8.4 wiederholen – und in Kapitel 9.12 findet sich dann die Lösungen mit Containern. 7.13 Lsg. zu Aufgabe „Zahlen-Liste“ – Kap. 7.9.4 Im ersten Augenblick sieht die Aufgabe ganz einfach aus: eine Klasse mit Main-Funktion und For-Schleife mit Zahlen-Ausgabe und die Klammern-Ausgabe vor und nach der For-Schleife – quasi analog zur Aufgabe 7.9.1 mit der Lösung 7.10. // Achtung – fehlerhafte Loesung public static void main(String[] args) { System.out.print('('); for (int i=1; i<=5; i++) { System.out.print(i + '-'); } System.out.println(')'); } Ausgabe: (1-2-3-4-5-) Leider ist diese Lösung falsch, denn hinter der letzten Zahl (hier „5“) folgt noch ein Bindestrich. Das Problem ist ein ganz typisches Alltags-Problem der Programmierung: beim ersten oder letzten Durchlauf einer Schleife müssen Dinge anders gehandelt werden. Schauen wir uns die drei typischen Lösungen an: • Bedingte Ausführung in der Schleife • Schleife um einen Durchlauf kürzen • Schleife mit Abbruch in der Mitte 7.13.1 Lösung 1: Bedingte Ausführung in der Schleife Der erste Ansatz ist einfach, in der Schleife mit einer If-Anweisung den letzten Durchlauf gesondert zu behandeln, z.B. so: for (int i=1; i<=5; i++) { System.out.print(i); if (i!=5) { System.out.print('-'); } } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 129 / 409 Frage: was gefällt ihnen an diesem Quelltext nicht? Ich hoffe, dass ihnen sofort unangenehm aufgestoßen ist, dass die Konstante „5“ zweimal im Quelltext auftaucht. Bei Änderungen müssen also zwei Stellen im Quelltext geändert werden. Nun sind die beiden Änderungen nicht viel Arbeit – das Problem ist in der Praxis, dass eine Stelle vergessen wird. Ein ganz wichtiger Grundsatz beim Programmieren ist, dass eine Änderungswunsch an der Software (hier: „Lass die Liste bis 8 gehen“) nur zu Änderungen an einer Stelle im Code führen sollte. Statt des Literals „5“ sollte also lieber eine Konstante benutzt werden. Da wir Konstanten noch nicht kennen, wäre die hier angemessene Lösung eine Variable. Um aus einer Variablen eine Konstante zu machen, muss man aber nur den Modifier „final“ verwenden (siehe Kapitel 8.3) – das können wir hier ruhig schon direkt benutzen. final int limit = 5; for (int i=1; i<=limit; i++) { System.out.print(i); if (i!=limit) { System.out.print('-'); } Ansonsten kann man die Lösung gut nutzen, und ist mit Klasse, Main-Funktion und der Ausgabe der Klammern schnell fertig: public class Appl { public static void main(String[] args) { final int limit = 5; System.out.print('('); for (int i=1; i<=limit; i++) { System.out.print(i); if (i!=limit) { System.out.print('-'); } } System.out.println(')'); } } Nachteil dieser Lösung ist, dass die Schleife durch die If-Anweisung etwas komplizierter und langsamer ist. 7.13.2 Lösung 2: Schleife um einen Durchlauf kürzen Da hier die Anzahl an Durchläufen festliegt, kann man die Schleife einfach um einen Durchlauf kürzen, und die spezielle Behandlung der letzten Zahl hinter der Schleife ganz für sich machen: public class Appl { public static void main(String[] args) { final int limit = 5; System.out.print('('); for (int i=1; i<=limit-1; i++) { System.out.print(i + '-'); } System.out.print(limit); System.out.println(')'); // <- hier steht jetzt "limit-1" } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 130 / 409 } Der Vorteil der Lösung ist, dass die Schleife ohne If-Anweisung einfacher und performanter ist. Leider gibt es Situationen, in denen diese Lösung nicht anwendbar ist, z.B. wenn die SchleifenAbbruch-Bedingung erst innerhalb der Schleife berechnet werden kann. Außerdem ist hier nachteilig, dass wir genau genommen eine Code-Verdopplung haben. Es fällt in unserem kleinen Beispiel gar nicht so richtig auf, aber die Ausgabe der Zahl findet sich jetzt in und nach der Schleife. Im Prinzip sieht unser Programm jetzt ja so aus: Schleife { tue-a tue-b } tue-a In einem echten Programm könnte „tue-a“ viel komplizierter Code sein, der dann doppelt vorhanden wäre. Damit haben wir wieder ein potentielles Problem im Änderungsfall (Fehler, Wartung, Refactoring, Anforderungen). Also wenn man diese Lösung wäht, dann sollte man „tue-a“ auf jeden Fall in eine extra Funktion auslagern. 7.13.3 Lösung 3: Schleife mit Abbruch in der Mitte Eine andere Lösung wäre die Schleife mittendrin zu verlassen: Schleife { tue-a Abbruch-wenn-noetig tue-b } Mit diesem Muster läßt sich die Code-Verdopplung beseitigen, und diese Lösung funktioniert auch dann wenn die Abbruch-Bedingung erst in der Schleife berechnet werden kann. Leider enthält auch sie noch die If-Anweisung, und ist daher nicht ganz so einfach und schnell wie die zweite Lösung. public class Appl { public static void main(String[] args) { final int limit = 5; System.out.print('('); for (int i=1; ; i++) { System.out.print(i); if (i==limit) { break; } System.out.print('-'); } System.out.println(')'); } } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 131 / 409 7.14 Lsg. zu Aufgabe „Mittelwert und Varianz“ – Kap. 7.9.5 Erklärung folgt (hoffentlich) später – todo... public class Appl { public static void main(String[] args) { if (args.length==0) { System.out.println("Keine Werte => keine Berechnungen"); return; } if (args.length==1) { try { double d = Double.parseDouble(args[0]); System.out.println("Nur ein Wert"); System.out.println("=> Mittelwert: " + d); System.out.println("=> Varianz: ---"); } catch (Exception x) { System.out.println("Fehlerhafter Parameter"); } return; } int count = args.length; double sum = 0.0; double squares = 0.0; for (int i=0; i<count; i++) { try { double d = Double.parseDouble(args[i]); sum += d; squares += d*d; } catch (Exception x) { System.out.println(i+1 + "-ter Parameter \"" + args[i] + "\" fehlerhaft"); return; } } System.out.println(count + " Werte"); System.out.println("=> Mittelwert: " + (sum/count)); System.out.println("=> Varianz: " + (squares/(count-1))); } } import java.io.BufferedReader; import java.io.InputStreamReader; public class Appl { public static void main(String[] args) { InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); int count = 0; double sum = 0.0; double squares = 0.0; while (true) { try { System.out.print("Wert: "); String in = reader.readLine(); if (in.length()==0) break; double d = Double.parseDouble(in); count++; sum += d; squares += d*d; } catch (Exception x) { System.out.println("Fehlerhafte Eingabe, bitte wiederholen"); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 132 / 409 } } if (count==0) { System.out.println("Keine Werte => keine Berechnungen"); return; } if (count==1) { System.out.println("Nur ein Wert"); System.out.println("=> Mittelwert: " + sum); System.out.println("=> Varianz: ---"); return; } System.out.println(count + " Werte"); System.out.println("=> Mittelwert: " + (sum/count)); System.out.println("=> Varianz: " + (squares/(count-1))); } } 7.15 Lsg. zu Aufgabe „Multiplikations-Matrix“ – Kap. 7.9.6 Erklärung folgt (hoffentlich) später – todo... public class Appl { public static void main(String[] args) { if (args.length!=2) { System.out.println("Das Programm erwartet zwei Int-Argumente"); return; } int v1; try { v1 = Integer.parseInt(args[0]); } catch (NumberFormatException x) { System.out.println("Argument 1 ist kein Int-Wert"); return; } int v2; try { v2 = Integer.parseInt(args[1]); } catch (NumberFormatException x) { System.out.println("Argument 2 ist kein Int-Wert"); return; } if (v1>v2) { int temp = v1; v1 = v2; v2 = temp; } // Zahlenbereich auf akzeptable Groesse checken if (v2-v1>=10) { System.out.println("Zu grosse Matrix"); return; } // Und Ausgabe... System.out.print(" |"); for (int i=v1; i<=v2; i++) { System.out.print(" " + i); } System.out.println(); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 133 / 409 System.out.print("---+"); for (int i=v1; i<=v2; i++) { System.out.print("---");; } System.out.println('-'); for (int i=v1; i<=v2; i++) { System.out.print(' ' + i + " |"); for (int j=v1; j<=v2; j++) { System.out.print(" " + i*j); } System.out.println(); } } } import java.io.BufferedReader; import java.io.InputStreamReader; public class Appl { public static void main(String[] args) { InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); int v1; try { System.out.print("Untere Grenze: "); String in = reader.readLine(); v1 = Integer.parseInt(in); } catch (NumberFormatException x) { System.out.println("Fehlerhafte Eingabe"); return; } int v2; try { System.out.print("Obere Grenze: "); String in = reader.readLine(); v2 = Integer.parseInt(in); } catch (NumberFormatException x) { System.out.println("Fehlerhafte Eingabe"); return; } if (v1>v2) { int temp = v1; v1 = v2; v2 = temp; } // Zahlenbereich auf akzeptable Groesse checken if (v2-v1>=10) { System.out.println("Zu grosse Matrix"); return; } // Und Ausgabe... System.out.print(" |"); for (int i=v1; i<=v2; i++) { System.out.print(" " + i); } System.out.println(); System.out.print("---+"); for (int i=v1; i<=v2; i++) { System.out.print("---");; } System.out.println('-'); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 134 / 409 for (int i=v1; i<=v2; i++) { System.out.print(' ' + i + " |"); for (int j=v1; j<=v2; j++) { System.out.print(" " + i*j); } System.out.println(); } } } 8 Funktionen In Java gibt es keine freien (globalen) Funktionen – alle Funktionen sind immer einer Klasse zugeordnet. Dies gilt auch für die Main-Funktionen. Syntax: [Modifizierer] <Rückgabetyp> <Fkt-Name> ( [Parameterliste] ) { <Implementierung> } • Modifizierer – z.B.: public, private, static, final, ... • Eine Parameterliste ist eine durch Komma getrennte Auflistung von Parametern (sie kann auch leer sein. Ein Parameter besteht aus Typ und Name. Bsp: public static void f() { ... } protected final int doit(StringBuffer sb) { ... } private static String calcName(String s, int i) { ... } Aufruf: <Fkt-Name>( <Argumentliste> ); Bsp: f(); calcName("", 1); Rückgaben können ignoriert werden – siehe Bsp „calcName“. Bemerkung – fast jeder ausführbare Code steht immer in einer Funktion. Die einzigen Ausnahmen sind Initialisierungs-Blöcke, die aber im Prinzip auch nur sehr spezielle Funktionen sind. Bemerkung – Funktions-Namen werden in Java per Konvention klein begonnen, und dann kapitalisiert fortgesetzt – vergleiche z.B. den Namen „calcName“. Siehe auch Kapitel 3.2.5. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 135 / 409 8.1 Funktions-Arten Achtung – in Java gibt es zwei Arten von Funktionen: • Element-Funktionen • Klassen-Funktionen Klassen-Funktionen sind Funktionen, die mit dem Modifizierer „static“ definiert sind – wie z.B. die Funktion „main“ in unseren Beispielen. Nur Klassen-Funktionen können direkt aufgerufen werden. Element-Funktion benötigen einen Objekt-Bezug, und werden dann mit dem Objekt und dem Punkt-Operator „.“ aufgerufen. Für Objekte z.B. vom Typ „String“ benutzen wir die Element-Funktionen einfach. Im Detail kennen lernen werden wir sie erst mit der Einführung von Klassen und Objekten in Kapitel todo. Solange sollten Sie in ihren Beispielen nur Klassen-Funktionen definieren, d.h. jede selbstgeschriebene Funktion sollte den Modifizierer „static“ bekommen. public class Appl { public static void sf() { ... } public void f() { ... } public static void main(String[] args) { sf(); f(); String s = "Hallo"; s.length(); } // Aufruf geht, da sf static ist // Compiler-Fehler, da Element-Fkt // Okay – Aufruf von Element-Fkt // mit Objekt-Bezug } 8.2 Syntaktischer Aufbau Eine Java Funktion besteht aus einer Folge von Anweisungen. Eine Anweisung ist: • eine Instruktion (wird durch Semikolon beendet – z.B. Zuweisung, Definition., Funktionsaufruf,...), • eine Kontrollstruktur (if, while, switch,...), oder • ein Block. Ein Block ist eine Folge von Anweisungen, die durch die geschweiften Klammern eingeschlossen sind – ein Block ist immer auch ein Scope, d.h. ein Sichtbarkeitsbereich von Variablen (siehe Kapitel 8.3). 8.3 Variablen in Funktionen Variablen können an (fast) jeder beliebigen Stelle in Funktionen definiert werden. Dabei werden © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 136 / 409 sogenannte lokale Variablen erzeugt, die lokal zur Funktion sind31. void fct() { int var = 23; System.out.println(var); } Ein einmal vergebener Name darf im gleichen Block nicht ein zweites Mal benutzt werden – ansonsten wäre der Name ja nicht eindeutig. void fct() { int var = 23; System.out.println(var); int var = 42; } // Compiler-Fehler – var schon definiert Wird eine Variable innerhalb eines Blocks in einer Funktion definiert, so ist sie auch nur innerhalb dieses Blocks sichtbar. Ein Block bildet also einen Sichtbarkeitsbereich. void fct() { while (true) { int var = 87; System.out.println(var); break; } System.out.println(var); } // Okay – var ist sichtbar // Compiler-Fehler – var hier nicht sichtbar Eine Besonderheit sind hierbei Variablen-Definitionen im Initialisierungs-Teil einer For-Schleife. Obwohl sie nicht in einem speziellen Block definiert sind, sind diese Variablen nur in der Schleife bekannt – siehe Kapitel 7.3. Ist eine lokale Variabel sichtbar (d.h. sie befinden sich innerhalb des definierenden Scopes in der Funktion), so darf auch in verschachtelten Blöcken keine weitere Variable des gleichen Names definiert werden.32 int n = 4; { int n = 3; } // Compiler-Fehler – Variable n ist schon definiert. 8.3.1 Modifier „final“ für lokale Variablen Der einzig erlaubte Modifizierer für Variablen in Funktionen ist der Modifizierer „final“. Wird er angegeben, so kann die Variable nicht verändert werden. final int n = 3; n = 4; // Compiler-Fehler – Variable darf nicht geaendert werden final StringBuffer sb = new StringBuffer("StringBuffer"); Genau genommen sind die Variablen nicht nur lokal zur Funktion, sondern sogar lokal zum Funktions-Aufruf. Lokale Variablen werden für jeden Funktionsaufruf beim Durchlaufen der Definition neu erzeugt, d.h. lokale Variablen sind wiedereinsprungsfest (reentrant). Wichtig ist dies für Rekursion (Kapitel 8.7) und Multithreading (nicht Thema des Tutorials). 32 Dies ist in Sprachen wie z.B. C oder C++ anders. 31 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 137 / 409 sb = new StringBuffer("Error"); // Compiler-Fehler – Variable ist final sb.append(" das geht aber"); // Okay, final schuetzt nur die Variable // nicht das referenzierte Objekt Achtung – bei Referenz-Variablen bezieht sich „final“ nur auf die Referenz, nicht auf das referenzierte Objekt – siehe auch das vorherige Beispiel. 8.4 Funktions-Parameter Funktions-Parameter sind eigentlich ganz normale lokale Variablen, die beim Funktions-Aufruf mit den Argumenten des Aufrufers initialisiert werden. Damit unterliegen sie natürlich auch der Wert- bzw. der Referenz-Semantik (siehe Kapitel 5.4), und das ist für Funktions-Parameter interessant. Alle elementaren Datentypen werden per Wert-Semantik, d.h. mit einer echten Kopie, übergeben. Änderungen innerhalb der Funktion haben nur Einfluss auf die Kopie in der Funktion, und betreffen das Original beim Aufrufer nicht. void f1(int n1) { System.out.println("=> System.out.println(" n1 += 12; System.out.println(" System.out.println("<= } f1"); " + n1); " + n1); f1"); int n = 23; System.out.println(n); f1(n); System.out.println(n); Ausgabe 23 => f1 23 35 <= f1 23 Anders ist dies dagegen bei Referenz-Variablen, d.h. bei allen Variablen, deren Typen keine elementaren Datentypen sind. In diesem Fall sind auch Funktions-Parameter ReferenzVariablen, und sie enthalten nur eine Referenz auf das eigentliche Objekt. Eine Funktion kann also das referenzierte Objekt des Aufrufers verändern! void f1(StringBuffer sb1) System.out.println("=> System.out.println(" sb1.append(" Welt"); System.out.println(" System.out.println("<= } { f1"); " + sb1); " + sb1); f1"); StringBuffer sb = new StringBuffer("Hallo"); System.out.println(sb); f1(sb); System.out.println(sb); Ausgabe © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 138 / 409 Hallo => f1 Hallo Hallo Welt <= f1 Hallo Welt Achtung – Sie können dieses Verhalten nicht beeinflussen33. Die Übergabe-Semantik des Funktions-Parameters liegt automatisch mit dem Typ des Parameters fest. 8.5 Funktions-Rückgaben Hat eine Funktion den Rückgabe-Typ „void“ – siehe auch Kapitel 5.2.1 – so gibt sie nichts zurück34. void f() { } Soll sie dagegen etwas zurückgeben, so muss der entsprechende Rückgabe-Typ in der Funktions-Definition angeben werden. Hat eine Funktion einen von „void“ verschiedenen Rückgabe-Typ, so muss sie mit einer Return-Anweisung beendet werden, die den RückgabeWert definiert. Der Rückgabe-Wert in der Return-Anweisung muss natürlich zum RückgabeTyp der Funktion passen. int f1() { return 12; } String f2() { return "Java"; } StringBuffer f3() { StringBuffer sb = new StringBuffer("Text"); return sb; } List<String> f4() { return null; } Eine Funktion kann beliebig viele Ausgänge haben. Jede Return-Anweisung beendet die Funktion instantan, und gibt den ausgewerteten Ausdruck der Return-Anweisung zurück. int sgn(int n) { if (n<0) { return –1; } return n==0 ? 0 : 1; } System.out.println("-6 System.out.println("-1 System.out.println(" 0 System.out.println(" 1 System.out.println(" 3 33 34 => => => => => " " " " " + + + + + sgn(-6)); sgn(-1)); sgn( 0)); sgn( 1)); sgn( 3)); Im Gegensatz zu z.B. C++. In manchen Sprachen werden solche Funktionen auch Prozeduren genannt. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 139 / 409 Ausgabe -6 => -1 -1 => -1 0 => 0 1 => 1 3 => 1 Eine Void-Funktion, d.h. eine Funktion ohne Rückgabe, kann jederzeit mit einer leeren ReturnAnweisung beendet werden. void f(int i) { if (i<12) { return; } System.out.println(i); } // Leere Return-Anweisung Funktionen können jeden beliebigen Typ zurückgeben, sowohl elementare als auch benutzerdefinierte Typen. Hierbei unterliegen natürlich auch die Funktions-Rückgaben der entsprechenden Wert- bzw. Referenz-Semantik. Funktions-Rückgaben können direkt benutzt werden, d.h. werden Objekte wie z.B. Strings zurückgegeben, so können für sie direkt Element-Funktionen aufgerufen werden. Man nennt dies auch Funktions-Verkettung. String fct() { return "Java"; } int len1 = fct().length(); System.out.println(len1); // Aufruf der Funktion 'length()' fuer den // zurueckgegebenen String String s = fct(); int len2 = s.length(); System.out.println(len2); // So haette man es auch machen koennen // Mit lokaler Zwischen-Variable 's' Ausgabe 4 4 8.6 Überladen Funktionen können überladen werden, d.h. der Funktions-Name darf mehrfach benutzt werden, solange sich die Funktionen durch ihre Parameter-Typen unterscheiden, d.h. der Compiler den Aufruf eindeutig zuordnen kann. Möglich ist dabei auch die Variation der Anzahl an Parametern – auch dort ist die Funktion exakt erkennbar. Der Rückgabetyp trägt nicht zur Unterscheidung bei. public class A { public static void f() { System.out.println("---"); } public static void f(int arg) { System.out.println("int: " + arg); } public static void f(String arg) { © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 140 / 409 System.out.println("String: " + arg); } public static void f(int arg1, String arg2) { System.out.println("int, String: " + arg1 + ", " + arg2); } public static void main(String[] args) { f(); f(8); f("Willy"); f("42"); // Ist natuerlich auch ein String, kein 'int' f(3, "Java"); } } Ausgabe --int: 8 String: Willy String: 42 int, String: 3, Java Hinweis – das Feature „Überladen“ funktioniert sowohl mit Klassen-Funktionen (Kapitel 12.4.4), mit Element-Funktionen (Kapitel 11.4.4), als auch mit Konstruktoren (Kapitel 11.4.7). 8.7 Rekursion Wenn eine Funktion im gleichen Thread mehrfach ineinander (d.h. nicht nacheinander, sondern gleichzeitig parallel) aufgerufen wird, nennt man das „Rekursion“ bzw. „rekursive Programmierung“. Eine Funktion muss sich dabei nicht zwangsläufig selber aufrufen, sondern dies kann auch über mehrere Zwischenstationen passieren, z.B. „f() => g() => h() => f()“. Solche Fälle sind häufig gar nicht mehr sofort zu erkennen, von daher passiert dies in der Praxis häufiger, als man vielleicht im ersten Augenblick denkt. Es gibt aber auch viele Probleme, die sich rekursiv viel viel leichter programmieren lassen als ohne Rekursion. Hier ein Beispiel, das man in der Praxis sicher nicht rekursiv lösen würde – die Summe aller Zahlen von 1 bis n. public static long sum(long arg) { if (arg<=1) { // Operator <= statt == um die Funktion bei fehlerhaften return 1; // Argumenten (arg<1) sauber zu beenden. } long erg = arg + sum(arg-1); return erg; } Hierbei wird quasi direkt die mathematische Definition umgesetzt: sum(1) := 1 sum(n) := n + sum(n-1) Natürlich läßt sich die Summe der Zahlen von 1 bis n direkt über die Gaußsche-Summen- © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 141 / 409 Formel „n*(n+1)/2“ berechnen – was hier auch viel sinnvoller wäre – aber es geht eben auch rekursiv. Hinweis – es läßt sich übrigens beweisen, dass jedes Problem was iterativ gelöst werden kann (d.h. mit einer Schleife) sich auch rekursiv lösen läßt, und umgekehrt. Praxis – in den aller-meisten Fällen sind die iterativen Lösungen schneller und benötigen weniger Speicher – sie sind daher vorzuziehen. Ein Schleifen-Durchlauf ist schnell und effizient, während ein Funktions-Aufruf doch relativ teuer ist (bezogen auf die Performance). In so manchen Fällen ist die rekursive Lösung aber ein 5-Zeiler, während die iterative Lösung Tage harter Arbeit sein kann und hinterher aus vielen Quelltext-Zeilen besteht – Beispiele für Lösungen, die rekursive erstaunlich einfach sind, finden sich z.B. in Kapitel 9.7.1 („Rekursive Datei-Suche“) und in der Musterlösung 12.9 zur Aufgabe 12.8.1 („Türme von Hanoi“). In einigen der folgenden Aufgaben werden sowohl iterative als auch rekursive Lösung verlangt. Die Lösungen bieten also Diskussions-Material für das Thema Performance und Implementierbarkeit. 8.8 Aufgaben 8.8.1 Aufgabe „Fakultäts Funktion“ Schreiben Sie zwei Versionen der Fakultäts-Funktion. Die Fakultät ist das Produkt der natürlichen Zahlen von 1 bis n. Mathematisch sieht die Definition folgendermaßen aus: • fak(0) :== 1 • fak(n) :== n * fak(n-1) Version 1 soll das Ergebnis iterativ berechnen – d.h. mit einer Schleife. Version 2 soll das Ergebnis rekursiv berechnen – d.h. durch Aufruf von sich selber. Welche Version würden Sie in der Praxis warum vorziehen? Schreiben Sie zusätzlich ein kleines Haupt-Programm, dass beide Funktionen für einige IntWerte benutzt. Achtung – bedenken Sie, dass die Fakultät sehr schnell wächst, und schon „14!“ nicht mehr in einen vorzeichenbehafteten 32-Bit Integer-Wert paßt, und „21!“ auch eine Long-Wert sprengt. Lösung siehe Kapitel 8.9. 8.8.2 Aufgabe „Primzahl?“ Schreiben Sie eine Funktion, die eine positive Integer-Zahl erwartet und zurückgibt ob die Zahl eine Primzahl ist. Schreiben Sie ein Programm, dass diese Funktion nutzt. Das Programm soll sowohl mit einem Kommandozeilen-Argument umgehen können, als auch interaktiv mit dem Benutzer die Zahl einlesen können. Hierbei gilt folgende Strategie: © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 142 / 409 • Sind Kommandozeilen Argumente vorhanden, so werden Sie alle für den Primzahl-Check genommen. • Ist kein Kommandozeilen Argument vorhanden, so wird eine Zahl interaktiv vom Benutzer erfragt. Lösung siehe Kapitel 8.10. 8.8.3 Aufgabe „Schleife rekursiv“ Nachdem Sie nun Funktionen und Rekursion kennen gelernt haben, schreiben Sie die einfachen Zahlen-Schleife aus Aufgabe 7.9.1 mit einer rekursiven Funktion statt mit einer Schleife. Ausgabe 1 2 3 4 5 Lösung siehe Kapitel todo. 8.8.4 Aufgabe „Zahlen-Liste 2“ Und auch die Zahlen-Liste aus Aufgabe 7.9.4 läßt sich nun mit Funktionen und Rekursion neu schreiben. Ausgabe: (1-2-3-4-5) Lösung siehe Kapitel todo. 8.8.5 Aufgabe „Quadratwurzel“ Schreiben Sie ein Programm, das die Quadratwurzel einer Fließkomma-Zahl größer-gleich 1.0 berechnet. Benutzen Sie hierbei nur die Grundrechenarten und Vergleiche, indem Sie sich der Lösung durch Intervall-Schachtelung Schritt für Schritt annähern. Das Konzept der Intervall-Schachtelung arbeitet folgendermaßen: • Beginnen Sie mit zwei Zahlen, von denen Sie ausgehen können, dass sie kleiner-gleich bzw. größer-gleich der Quadratwurzel sind. • Berechnen Sie den Mittelwert der beiden Zahlen, und quadrieren Sie diesen. • Weicht der quadrierte Mittelwert um weniger als einen Toleranzwert epsilon (z.B. 1.0e-5) von der Eingabe-Zahl ab, so verwenden Sie den Mittelwert als Ergebnis der Berechnung. D.h. Sie haben die Quadratwurzel gefunden. • Ist das Quadrat des Mittelwertes größer als die Eingabe-Zahl, so verwenden Sie das untere Intervall für die nächste Iteration. • Ist das Quadrat des Mittelwertes kleiner als die Eingabe-Zahl, so verwenden Sie das obere Intervall für die nächste Iteration. • Führen Sie die Iteration so lange durch, bis sich der quadrierte Mittelwert bis auf den © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 143 / 409 Toleranzwert epsilon an die Eingabe-Zahl angenähert hat. Geben Sie am Anfang der Berechnung den Toleranzwert epsilon aus. Geben Sie vor jeder Iteration die Nummer der Iteration und das aktuelle Intervall aus. Und geben Sie am Ende natürlich auch die Wurzel aus. Mögliche Ein- und Ausgabe Wurzel-Berechnung fuer Zahlen groesser-gleich 1.0 Zahl: 4 Berechne Wurzel aus 4 mit einem Epsilon von 0.0001 - Durchlauf 1: 1 < sqrt(4) < 4 - Durchlauf 2: 1 < sqrt(4) < 2.5 - Durchlauf 3: 1.75 < sqrt(4) < 2.5 - Durchlauf 4: 1.75 < sqrt(4) < 2.125 - Durchlauf 5: 1.9375 < sqrt(4) < 2.125 - Durchlauf 6: 1.9375 < sqrt(4) < 2.03125 - Durchlauf 7: 1.98438 < sqrt(4) < 2.03125 - Durchlauf 8: 1.98438 < sqrt(4) < 2.00781 - Durchlauf 9: 1.99609 < sqrt(4) < 2.00781 - Durchlauf 10: 1.99609 < sqrt(4) < 2.00195 - Durchlauf 11: 1.99902 < sqrt(4) < 2.00195 - Durchlauf 12: 1.99902 < sqrt(4) < 2.00049 - Durchlauf 13: 1.99976 < sqrt(4) < 2.00049 - Durchlauf 14: 1.99976 < sqrt(4) < 2.00012 - Durchlauf 15: 1.99994 < sqrt(4) < 2.00012 - Durchlauf 16: 1.99994 < sqrt(4) < 2.00003 Wurzel: 1.99998 Gibt es mehrere Arten, wie sie diese Funktion implementieren können? Wenn ja, dann machen Sie das auch. Lösung siehe Kapitel 8.13. 8.9 Lsg. zu Aufgabe „Fakultäts Funktion“ – Kap. 8.8.1 8.9.1 Iterative Lösung Die iterative Berechnung der Fakultät, d.h. mit einer Schleife, läß sich im Prinzip direkt runter implementieren. Man muss ja nur alle Zahlen von „0“ bis „n“ inkl. aufmultiplizieren, und dass Ergebnis zurückgeben. public static long facIterative(int n) { long res = 1; for (int i=2; i<=n; i++) { res *= i; } return res; } Hinweis – da die Fakultät von „0“ und „1“ selber „1“ sind, braucht die Schleife erst ab „2“ los zulaufen. Falls die Funktion mit dem Argument „0“ oder „1“ aufgerufen wird, ist dies auch kein Problem, da eine For-Schleife kopfgesteuert ist, d.h. auch kein Mal durchlaufen werden kann. Hinweis – passen Sie auf, dass Sie die Schleife bis „n“ inkl. laufen lassen, d.h. die AbbruchBedingung ist „i<=n“ mit dem Operator „kleiner-gleich“. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 144 / 409 8.9.2 Rekursive Lösung Auch die rekusive Berechnung ist nicht weiter schwierig, da man hier quasi direkt die mathematische Definition abschreiben kann: • fak(0) :== 1 • fak(1) :== 1 • fak(n) :== n * fak(n-1) public static long facRecursive(int n) { if (n<=1) { return 1; } return n * facRecursive (n-1); } 8.9.3 Main-Funktion Die Aufgabe fordert auch eine kleine Main-Funktion. Hier eine, die beide Funktionen parallel nutzt, und den Benutzer ihre Ergebnisse vergleichen läßt. public static void main(String[] args) { for (int i=0; i<=20; i++) { long fi = facIterative(i); long fr = facRecursive(i); System.out.println(i + "! => " + fi + " } } " + fr); Ausgabe 0! => 1 1 1! => 1 1 2! => 2 2 3! => 6 6 4! => 24 24 5! => 120 120 6! => 720 720 7! => 5040 5040 8! => 40320 40320 9! => 362880 362880 10! => 3628800 3628800 11! => 39916800 39916800 12! => 479001600 479001600 13! => 6227020800 6227020800 14! => 87178291200 87178291200 15! => 1307674368000 1307674368000 16! => 20922789888000 20922789888000 17! => 355687428096000 355687428096000 18! => 6402373705728000 6402373705728000 19! => 121645100408832000 121645100408832000 20! => 2432902008176640000 2432902008176640000 Und welche Lösung sollte man in der Praxis benutzen? Hier ist sicher die iterative Lösung vorzuziehen, die vollkommen unproblematisch zu implementieren ist, und vor allem bzgl. Performance vorzuziehen ist, da ein Schleifen-Durchlauf im Normall-Fall einiges billiger ist als ein Funktions-Aufruf. Achtung – keine der Lösung kümmert sich wirklich gut um das Problem: „Was passiert, wenn © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 145 / 409 die Funktion mit einer negativen Zahl aufgerufen wird?“ Beide Lösungen geben dann „1“ zurück, was zwar ein sauber definiertes Verhalten ist, aber auch sicher nicht das richtige Ergebnis. Die typische Java-Lösung, wie wir sie z.B. schon aus den Kapiteln 3.5, 3.10 oder 3.11 kennen, ist die Benutzung von Exceptions – siehe Kapitel todo. Da wir Exceptions noch nicht kennen, ignorieren wir dieses Problem hier erstmal (sollten es aber nicht vergessen). 8.9.4 Gesamter Quelltext Hier noch mal der ganze Quelltext inkl. Klasse. public class Appl { public static long facIterative(int n) { long res = 1; for (int i=2; i<=n; i++) { res *= i; } return res; } public static long facRecursive(int n) { if (n<=1) { return 1; } return n*facRecursive(n-1); } public static void main(String[] args) { for (int i=0; i<=20; i++) { long fi = facIterative(i); long fr = facRecursive(i); System.out.println(i + "! => " + fi + " } } " + fr); } 8.10 Lsg. zu Aufgabe „Primzahl?“ – Kap. 8.8.2 Erklärung folgt (hoffentlich) später – todo... import java.io.BufferedReader; import java.io.InputStreamReader; public class Appl { public static boolean prim(int n) { for (int i=2; i<=Math.sqrt(n); i++) { if (n%i==0) { return false; } } return true; } public static void check(String s) { try { int n = Integer.parseInt(s); if (n<0) { System.out.println("Negative Zahl: " + n + " - no check"); } else { System.out.println(n + " prim? -> " + prim(n)); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 146 / 409 } } catch (Exception x) { System.out.println("Keine Zahl: \"" + s + "\" - no check"); } } public static void main(String[] args) { if (args.length>0) { for (int i=0; i<args.length; i++) { check(args[i]); } return; } InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); try { System.out.print("Bitte geben Sie die zu checkende Zahl ein: "); String in = reader.readLine(); check(in); } catch (Exception x) { System.out.println("Probleme beim Einlesen"); } } } 8.11 Lsg. zu „Aufgabe „Schleife rekursiv“ – Kap. 8.8.3 Erklärung folgt (hoffentlich) später – todo... public class Appl { public static void loop(int n, int limit) { System.out.print(n + " "); if (n == limit) { return; } loop(n + 1, limit); } public static void main(String[] args) { loop(1, 5); } } 8.12 Lsg. zu Aufgabe „Zahlen-Liste 2“ – Kap. 8.8.4 Erklärung folgt (hoffentlich) später – todo... public class Appl { public static void loop(int n, int limit) { System.out.print('('); internalLoop(n, limit); System.out.print(')'); } public static void internalLoop(int n, int limit) { if (n == limit) { System.out.print(n); return; } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 147 / 409 System.out.print(n + "-"); internalLoop(n + 1, limit); } public static void main(String[] args) { loop(1, 5); } } 8.13 Lsg. zu Aufgabe „Quadratwurzel“ – Kap. 8.8.5 8.13.1 Lösung 1 - Iterativ Fangen wir mit den einfachen Dingen an: • Zuerst muss ein String von der Kommando-Zeile eingelesen werden. • Der muss in eine Fließkomma-Zahl gewandelt werden. • Diese Fließkomman-Zahl soll größer-gleich 1.0 sein. • Macht eine dieser Bereiche Probleme, beenden wir das Programm mit einer FehlerMeldung. • Die eigentliche Wurzel-Berechnung lagern wir in die Funktion „squareRoot“ aus. InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); try { System.out.print("Bitte geben Sie eine Zahl >=1 ein: "); String in = reader.readLine(); try { double d = Double.parseDouble(in); if (d<1.0) { System.out.println("Zahl: " + d + " zu klein"); } else { squareRoot(d); } } catch (Exception x) { System.out.println("Keine Zahl: \"" + s + "\""); } } catch (Exception x) { System.out.println("Probleme beim Einlesen"); } Ich hoffe, Sie sehen es wie ich: der Code sieht kompliziert aus. Man könnte natürlich die verschachtelten Try-Catch-Blöcke zu einem zusammenfassen, aber dann würden die FehlerMeldungen nicht mehr dediziert für das Problem sein35. Besser ist, wir trennen den Code durch Benutzung von Funktionen. public static void main(String[] args) { InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); try { System.out.print("Bitte geben Sie eine Zahl >=1 ein: "); String in = reader.readLine(); calc(in); } catch (Exception x) { System.out.println("Probleme beim Einlesen"); Okay, ich gebe zu: das könnte man hier trotzdem hinbekommen, da unterschiedliche Exceptions geworfen werden. Aber das lernen wir ja erst viel später in Kapitel todo. 35 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 148 / 409 } } public static void calc(String s) { try { double d = Double.parseDouble(s); if (d<1.0) { System.out.println("Zahl: " + d + " zu klein"); } else { squareRoot(d); } } catch (Exception x) { System.out.println("Keine Zahl: \"" + s + "\""); } } Das ganz hat jetzt außerdem den Vorteil, dass wir unser Programm problemlos komfortabler gestalten können – viel komfortabler als die Aufgabe das von uns erwartet. Aber wir machen ja gerne Sternchen-Aufgaben, da das mit Java so einfach ist und so viel Spaß macht. Schön wäre es, dass Programm sowohl mit einem Kommandozeilen-Argument als auch mit einer interaktiven Eingabe von der Kommandozeile benutzen zu können. Die Idee ist, wenn der Benutzer ein Kommandozeilen-Argument mitgibt, dann wird von diesem die Wurzel berechnet. Ansonsten wird die Eingabe interaktiv von der Konsole eingelesen. Dies ist jetzt einfach zu implementieren, da wir nur die Kommandozeilen-Argumente auswerten müssen, und im Falle von einem vorhandenen dieses an die Calc-Funktion weiterreichen müssen36. public static void main(String[] args) { if (args.length==1) { calc(args[0]); return; } ... } Damit bleibt nur noch die eigentliche Funktion „squareRoot“ zur Berechnung der Wurzel zur Implementierung übrig: • Zuerst wird ein Toleranzwert epsilon benötigt, der die mindestens notwendige Näherung an die wahre Lösung beschreibt. Hierfür bietet sich natürlich eine Konstante (siehe Kapitel 8.3) an. Noch besser wäre sicher eine Klassen-Konstante, aber die lernen wir erst in Kapitel 12.5.2 kennen. • Und es muss eine Start-Ausgabe geben. Alles zusammen kann dann z.B. so aussehen: public static double squareRoot(double v) { final double epsilon = 0.0001; System.out.println("Berechne Wurzel aus " + v + Wenn sie noch mehr an Sternchen-Aufgaben interessiert sind, dann sorgen sie doch einfach dafür, dass wenn mehrere Kommandozeilen-Argumente übergeben werden, für alle die Wurzel berechnet wird. 36 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 149 / 409 " mit einem Epsilon von " + epsilon); ... } Für die Intervallschachtelung benötigen wir zwei Zahlen, bei denen die eine auf jeden Fall kleiner-gleich der Lösung ist, während die andere größer-gleich der Lösung sein muss. Die Aufgabe ist zufälligerweise so gestellt, dass der Eingabe-Wert „v“ größer-gleich 1.0 sein muss, d.h. die Lösung wird immer auch größer-gleich 1.0 sein, und immer kleiner-gleich der EingabeZahl „v“. Als untere und obere Grenze bieten sich also diese Werte an. Die Berechnung des Mittelwerts „average“ der beiden Grenzwerte, und seine Quadratur „av2“ sind dann kein Problem mehr. double double double double lowerLimit = 1.0; upper_Limit = v; average = (lower_limit + upperLimit) / 2; av2 = average*average; Die Ausgabe vor jeder Iteration verschieben wir erstmal auf später – wir dürfen sie nur nicht vergessen. Wenn der quadrierte Mittelwert „av2“ den Eingabe-Wert bis auf den Toleranzwert epsilon erreicht hat, dann ist der Mittelwert die Lösung. Ansonsten muss die Iteration fortgesetzt werden. Dies klingt nach einer Schleife mit entsprechender Abbruch-Bedingung. Da bleibt höchstens noch die Frage, ob der Mittelwert „average“ bei dem die Eingabe „v“ auf genau epsilon erreicht wird, auch schon eine Lösung ist oder nicht. Letztlich also die Frage, ob die Abbruch-Bedingung „>=“ und „<=“ oder nur „>“ und „<“ enthält. Die Aufgaben-Stellung ist hier nicht ganz exakt, ich habe mir für erstes entschieden. Die Schleife sieht also folgendermaßen aus. while ((av2-epsilon>=v) || (av2+epsilon<=v)) { ... } Für den Fall, dass die Lösung noch nicht erreicht wurde, muss das Intervall – wie in der Aufgabe beschrieben – neu gewählt werden while ((av2-epsilon>=v) || (av2+epsilon<=v)) { if (v<av2) { upperLimit = average; } else { lowerLimit = average; } ... } Und dann muss für das neu gewählte Intervall der Mittelwert und sein Quadrat neu berechnet werden. while ((av2-epsilon>=v) || (av2+epsilon<=v)) { if (v<av2) { upperLimit = average; } else { lowerLimit = average; © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 150 / 409 } average = (lowerLimit+upperLimit)/2; av2 = average*average; } Im Prinzip ist die Aufgabe jetzt schon fertig. Wir sehen hier aber mal wieder eine Verletzung des DRY-Prinzips37, da die beiden Berechnungen nun zweimal im Quelltext vorkommen: initial vor der Schleife, und am Ende der Schleife. Dies liegt daran, dass hier wieder mal eine Schleife mit Ausgang in der Mitte benötigt wird. Also ändern wir dies um. double lowerLimit = 1.0; double upper_Limit = v; double average; for (int count=0;;) { average = (lower_limit+upperLimit)/2; double av2 = average*average; if ((av2-epsilon<v) && (av2+epsilon>v)) break; if (v<av2) { upperLimit = average; continue; } lowerLimit = average; } Jetzt fehlt nur noch die Ausgabe während jeder Iteration, und am Ende die Ausgabe der Lösung. for (int count=0;;) { System.out.println("- Durchlauf " + ++count + ": " + lowerLimit + " < sqrt(" + v + ") < " + upperLimit); ... } System.out.println("Wurzel: " + average); Alles zusammen sieht dann z.B. so aus: import java.io.BufferedReader; import java.io.InputStreamReader; public class Appl { //-----------------------------------------------------------------------public static void main(String[] args) { if (args.length>0) { calc(args[0]); return; } InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); try { System.out.print("Bitte geben Sie eine Zahl >=1 ein: "); String in = reader.readLine(); calc(in); } catch (Exception x) { System.out.println("Probleme beim Einlesen"); } } //-----------------------------------------------------------------------public static void calc(String s) { 37 DRY – don’t repeat yourself. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 151 / 409 try { double d = Double.parseDouble(s); if (d<1.0) { System.out.println("Zahl: " + d + " zu klein"); } else { squareRoot(d); } } catch (Exception x) { System.out.println("Keine Zahl: \"" + s + "\""); } } //-----------------------------------------------------------------------public static double squareRoot(double v) { final double epsilon = 0.0001; System.out.println("Berechne Wurzel aus " + v + " mit einem Epsilon von " + epsilon); double lowerLimit = 1.0; double upper_Limit = v; double average; for (int count=0;;) { System.out.println("- Durchlauf " + ++count + ": " + lowerLimit + " < sqrt(" + v + ") < " + upperLimit); average = (lower_limit+upperLimit)/2; double av2 = average*average; if ((av2-epsilon<v) && (av2+epsilon>v)) break; if (v<av2) { upperLimit = average; continue; } lowerLimit = average; } System.out.println("Wurzel: " + average); return average; } } 8.13.2 Lösung 2 - Rekursiv Natürlich läßt sich auch die Quadratwurzel Berechnung mit Intervall-Schachtelung rekursiv berechnen. Genau genommen zwingt sich der rekursive Algorithmus bei der Überlegung, wie man die Funktion implementiert, geradezu auf. Denn was macht man bei der Intervall-Schachtelung? Man teilt das Intervall, in dem sich die Lösung befinden muss, in zwei gleich große Teile, und untersucht in welchem Teil die Lösung liegen muss. Für diesen Teil macht man wieder das gleiche, d.h. wendet die IntervallSchachtelung auf diesen Teil an. Usw. Sie sehen, das klingt stark rekursiv. Und die Abbruch Bedingung ist natürlich, wenn die Lösung bis auf epsilon erreicht wurde. import java.io.BufferedReader; import java.io.InputStreamReader; public class Appl { //------------------------------------------------------------------------public static void main(String[] args) { if (args.length>0) { calc(args[0]); return; © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 152 / 409 } InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); try { System.out.print("Bitte geben Sie eine Zahl >=1 ein: "); String in = reader.readLine(); calc(in); } catch (Exception x) { System.out.println("Probleme beim Einlesen"); } } //------------------------------------------------------------------------public static void calc(String s) { try { double d = Double.parseDouble(s); if (d<1.0) { System.out.println("Zahl: " + d + " zu kleine - keine Berechnung"); } else { squareRoot(d); } } catch (Exception x) { System.out.println("Keine Zahl: \"" + s + "\" - keine Berechnung"); } } //------------------------------------------------------------------------public static double squareRoot(double v) { System.out.println("Berechne Wurzel aus " + v + " mit einem Epsilon von 0.0001"); double sqrt = squareRoot(v, 1.0, v, 1); System.out.println("Wurzel: " + sqrt); return sqrt; } //------------------------------------------------------------------------public static double squareRoot( double v, double lowerLimit, double upperLimit, int count) { final double epsilon = 0.0001; System.out.println("- Durchlauf " + count + ": " + lowerLimit + " < sqrt(" + v + ") < " + upperLimit); double average = (lowerLimit+upperLimit)/2; double av2 = average*average; if ((av2-epsilon<v) && (av2+epsilon>v)) { return average; } if (v<av2) { return squareRoot(v, lowerLimit, average, count+1); } return squareRoot(v, average, upperLimit, count+1); } } 9 Ausgewählte Bibliotheks-Klassen Die Java Bibliothek (JDK 1.8) umfaßt ~217 Packages mit ~4240 Klassen – siehe Kapitel 2. Für viele Probleme stehen daher fertige Klassen zur Verfügung. Ein paar ganz allgemeine und sehr zentrale Klassen möchte ich hier ganz kurz vorstellen, damit wir sie in späteren Beispielen bzw. dem Praktikum benutzen können. Aber machen Sie sich klar, es sind nur ganz wenige aus dem großen Angebot der Java-Welt. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 153 / 409 9.1 Klasse String Mit der Klasse String werden konstante Unicode Zeichenketten repräsentiert. Ja, Sie haben richtig gelesen: „konstante Zeichenketten“. Strings können aus Performance-Gründen nicht verändert werden. Wollen Sie Zeichenketten modifizieren, so müssen Sie die Klassen StringBuilder oder StringBuffer (siehe Kapitel 9.2) verwenden. Hinweise: • Man nennt Strings, wie auch andere unveränderbare Objekte, auch „immutable“ Objekte. • Strings können einfach so benutzt werden, und benötigen kein „import“. Sie kommen aus dem Package „java.lang“, das immer automatisch zur Verfügung steht – Kapitel 13.3. 9.1.1 Initialisierung Da Zeichenketten in der Praxis sehr häufig vorkommen, sind in der Sprache Java einige Unterstützungen für Strings integriert, die über eine normale ‚Klasse‘ hinaus gehen. So können Strings neben dem expliziten Erstellen mit „new“ auch einfach mit Zeichenketten erzeugt werden38. public class Kap_09_01_Bsp_01_Strings { public static void main(String[] args) { String s1 = new String("Java"); // So muessen in Java eigentlich alle System.out.println(s1); // Objekte angelegt werden: mit "new" String s2 = "Hallo Welt"; System.out.println(s2); // Aber bei Strings geht es auch ohne // "new" - Nettigkeit der Sprache s1 = "Viel einfacher so"; System.out.println(s1); // Und das geht auch bei Zuweisungen s1 = new String("Aufwaendig"); System.out.println(s1); // Die normale Art fuer Nicht-Strings } } Ausgabe Java Hallo Welt Viel einfacher so Aufwaendig 9.1.2 Operator + Für die Klasse String ist in der Sprache Java der Operator + definiert. Auf zwei Strings angewandt ist das Ergebnis ein neuer String, der aus den beiden konkatinierten Operanden besteht. public class Kap_09_01_Bsp_02_StringAddition1 { Im Gegensatz zu Arrays (siehe Kapitel 10.1) geht dies jederzeit, und nicht nur bei der Variablen-Definitionen. 38 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 154 / 409 public static void main(String[] args) { String s1 = "Hallo"; String s2 = " Welt"; String s = s1 + s2; System.out.println(s); } } Ausgabe Hallo Welt Wie schon kurz in Kapitel 3.4 angesprochen: ist nur einer der Operanden ein String, so wird der andere implizit in einen String umgewandelt. Wir sehen dies im folgenden Quelltext: • in der Zeile (*) für einige elementare Datentypen („boolean“, „int“ und „long“), • in der Zeile (**) nochmal expliziter für ein „int“, und • in der Zeile (****) für ein AWT-Label. Wir werden GUI-Label noch in Kapitel 20.1 näher kennen lernen – hier steht es einfach für ein komplexes Objekt. Als Beispiel, dass wirklich jedes Objekt beim Operator + in einen String gewandelt wird. import java.awt.Label; public class Kap_09_01_Bsp_03_StringAddition2 { public static void main(String[] args) { String s = "Variablen: "; boolean b = true; int i = 12345; long l = 123456789012345L; s = s + b + "=" + false + " - i=" + i + " - l=" + l; System.out.println(s); int n = 42; String s1 = "" + n; String s2 = "" + Integer.toString(n); System.out.println(s1); System.out.println(s2); Label label = new Label("Beschriftung"); s1 = "" + label; s2 = label.toString(); System.out.println(s1); System.out.println(s2); // Zeile (*) // Zeile (**) // Zeile (***) // Zeile (****) // Zeile (*****) } } Ausgabe Variablen: true=false - i=12345 - l=123456789012345 42 42 java.awt.Label[label0,0,0,0x0,invalid,align=left,text=Beschriftung] java.awt.Label[label0,0,0,0x0,invalid,align=left,text=Beschriftung] Alternativ zu den impliziten Wandlungen beim Operator + in z.B. Zeile (**) und (****) sind auch die mehr expliziten Wandlungen wie hier in den Zeilen (***) und (*****) möglich. • Für alle elementaren Datentypen existieren Standard-Umwandlungen, die sich auch in den jeweiligen Wrapper-Klassen wiederfinden – siehe Kapitel 3.11 und 9.4. • Für alle anderen Typen kann die Elementfunktion „toString()“ benutzt werden, die für alle Klassen definiert ist – siehe auch Kapitel 14.11.1. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 155 / 409 Bitte beachten Sie: auch wenn im Quelltext des letzten Beispiels „s=...“ steht, so wird nicht der String verändert. Es ändert sich nur der Wert der Referenz-Variablen „s“, die nach der Zuweisung auf ein anderes String-Objekt verweist. Das originale String-Objekt mit dem Wert „Variablen: “ wird nicht verändert, sondern jetzt nur nicht mehr referenziert. String String s = "abc" s = s + "def" s: => verändert s: unverändert "abc" String String "abc" "def" => String "abcdef" Abb. 9-1 : Ein String-Objekt wird nicht verändert Hinweis – die Addition von Strings mit dem Plus-Operator ist nicht sehr effizient. Wie Sie an der obigen Abbildung sehen können, müssen bei dieser Operation neue Objekte angelegt und die Zeichen kopiert werden – was relativ viel Zeit kostet. Nutzen Sie aufgrund der Performance Vorteile bei String-Verkettungen die Klassen „StringBuilder“ und „StringBuffer“, siehe Kapitel 9.2. 9.1.3 Referenz-Semantik und Konstante Zeichenketten Da String eine Klasse und kein elementarer Datentyp ist, unterliegen String-Variablen natürlich der Referenz-Semantik (siehe Kapitel 5.4.2). Und die Klasse String repräsentiert konstante Zeichenketten. D.h. es gibt keine Möglichkeit einen String zu verändern. Alle Operationen auf Strings lassen diese unangetastet, bzw. geben einen neuen zurück – der Originalstring bleibt immer unverändert, String-Objekte sind immutable. Damit entfällt das typische Problem der Referenzsemantik, nämlich dass ein Objekt von woanders einfach geändert wird. Strings sind dagegen resistent, und darum an vielen Stellen unproblematischer – vergleiche z.B. Kapitel todo oder auch Kapitel todo. Achtung – umgekehrt gibt es bei Strings dadurch auch eine Falle: der Vergleichs-Operator „==“ führt auch bei Strings nur einen Identitäts-Vergleich durch, und keinen Wert-Vergleich. Leider verhält sich der Operator „==“ bei Strings aber manchmal so, als würde er einen WertVergleich durchführen – siehe auch Kapitel 6.2. Vergleichen Sie Strings immer mit „equals“, auch wenn der Operator „==“ scheinbar funktioniert. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 156 / 409 9.1.4 Schnittstelle Die Klasse String ist in java.lang definiert. Sie hat keine public Attribute, aber viele ElementFunktionen – hier eine kleine Auswahl: String() Konstruktor: erzeugt einen leeren String. String(char[ ]) Konstruktor: erzeugt einen String aus einem char-Array. String(StringBuffer) Konstruktor: erzeugt einen String aus einem String-Buffer. boolean equals(String) Gibt zurück, ob die beiden Strings wert-gleich sind. boolean equalsIgnoreCase (String anotherString) Vergleicht zwei String ohne Berücksichtigung von Groß- und KleinSchreibung. char charAt(int) Gibt das Zeichen an der spezifizierten Position zurück. int length() Gibt die Anzahl an Zeichen des Strings zurück, d.h. seine Länge. String trim() Gibt einen neuen String zurück, der dem alten ohne führende und folgende Whitespaces (nicht sichtbare Zeichen wie z.B. Leerzeichen oder Tabulator) entspricht. boolean startsWith(String) Gibt zurück, ob der String mit dem übergebenen beginnt. boolean endsWith(String) Gibt zurück, ob der String mit dem übergebenen aufhört. String substring (int beginIndex, int endIndex) Gibt eine Teil-String als neuen String zurück. Der Teil-String beginnt beim Zeichen mit dem Index „beginIndex“ inkl. und endet beim Zeichen mit dem Index „endIndex“ exkl. Bei fehlerhaften Indices wird eine Exception geworfen. Die komplette Schnittstelle der Klasse „String“ finden Sie in der offiziellen Java ReferenzDokumentation, die Sie bei Oracle herunterladen können – siehe Kapitel 4.1.1. 9.2 Klassen StringBuilder & StringBuffer Die Klassen StringBuilder und StringBuffer repräsentieren Unicode Zeichenketten, die verändert werden dürfen. Beide Klassen enthalten viele Funktionen zur Modifikation von Texten, z.B. die Element-Funktion „append“, die einen Text an das bestehende StringBuilderoder StringBuffer-Objekt anfügen. public class Kap_09_02_Bsp_01_StringBuilderUndBuffer { public static void main(String[] args) { StringBuilder sb1 = new StringBuilder("1: Append"); sb1.append(" mit"); sb1.append(" StringBuilder"); System.out.println(sb1); StringBuffer sb2 = new StringBuffer("2: Append"); sb2.append(" mit"); sb2.append(" StringBuffer"); System.out.println(sb2); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 157 / 409 } } Ausgabe 1: Append mit StringBuilder 2: Append mit StringBuffer Im Gegensatz zur String-Addition wird hierbei wirklich das Objekt verändert und kein Neues erzeugt. Dafür müssen StringBuilder- und StringBuffer-Objekte ganz normal explizit mit „new“ erzeugt werden. Abb. 9-2 : Ein StringBuilder-Objekt wird verändert Die Klasse „StringBuffer“ existiert seit dem JDK 1.0, während die Klasse „StringBuilder“ erst mit dem JDK 1.5 eingeführt. Beide Klassen haben dieselbe Schnittstelle. Im Gegensatz zu StringBuffer ist StringBuilder aber nicht multi-threading fest, dafür aber wesentlich schneller. Sie sollten also in single-threaded Anwendungen, oder in MT-unkritischen Situation (wie z.B. String-Manipulation innerhalb einer Funktion mit einer lokalen Variable) die Klasse „StringBuilder“ gegenüber „StringBuffer“ bevorzugen. Benötigen Sie dagegen eine MT feste Klasse, so führt kein Weg an „StringBuffer“ vorbei – auch wenn dies mit Performance-Einbußen verbunden ist. Hinweis – auch die Klassen StringBuilder und StringBuffer können einfach so benutzt werden, und benötigen kein „import“. Vergleichbar zur Klasse String kommen sie aus dem Package „java.lang“, das immer automatisch zur Verfügung steht – siehe Kapitel 13.3. 9.3 Container & Iteratoren Container sind Objekte, die andere Objekte aufnehmen können, diese verwalten, und dabei automatisch passend wachsen. Es gibt nicht den einen Container, der für alle Zwecke optimal geeignet ist, sondern jeder Container hat seine Vor- und Nachteile. Je nach Anwendung bzw. Anforderung ist mal der Eine, mal der Andere besser geeignet. Nähere Informationen hierzu finden Sie in Kapitel todo, und ausführlicher in vielen Büchern über „Algorithmen und Datenstrukturen“. Seit der ersten Java Version (JDK 1.0) sind einige grundlegende Container Bestandteil der Java Klassen-Bibliothek. Mit der Version Java 2 (JDK 1.2) wurden die Container stark überarbeitet und erweitert, und dann mit jeder JDK Version weiter verbessert. So ist heute ein sehr ordentliches Container-Framework Bestandteil der Java Bibliothek. 9.3.1 Einführung in Container & Iteratoren Ein typischer Container ist die „ArrayList“. Sie wird – wie jedes Objekt in Java – mit „new“ erzeugt und kann dann einfach genutzt werden. In spitzen Klammern geben wir den Typ der Elemente an, die in der ArrayList gespeichert werden sollen. Mit diesen spitzen Klammern © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 158 / 409 nutzen wir die seit JDK 1.5 vorhandenen Generics, um den Container typsicher zu machen. Im folgenden Beispiel wird eine ArrayList für Strings angelegt und mit 3 Strings gefüllt. Die aktuelle Anzahl an Elementen im Container kann man dann mit „size()“ abfragen: import java.util.ArrayList; public class Kap_09_03_Bsp_01_ContainerEinstieg { public static void main(String[] args) { ArrayList<String> al = new ArrayList<String>(); al.add("Hallo"); al.add("Java"); al.add("Kurs"); System.out.println("Array-List enthaelt " + al.size() + " Elemente"); } } Ausgabe Array-List enthaelt 3 Elemente Die Typisierung mit den spitzen Klammern bewirkt, dass keine Objekte falschen Typs in den Container eingefügt werden können39. ArrayList<String> l = new ArrayList<String>(); l.add("Java"); // Okay, "Java" ist ein String l.add(123); // Compiler-Fehler, kein String l.add(new StringBuilder()); // Compiler-Fehler, auch kein String Seit JDK 1.7 kann man sich die Erzeugung eines typisierten Objektes erleichtern – man benötigt die Angabe des Element-Typs in den spitzen Klammern nicht mehr, da der Compiler diesen aus dem Variablen-Typ ermitteln kann. import java.util.ArrayList; public class Kap_09_03_Bsp_02_ContainerEinstiegJdk17 { public static void main(String[] args) { ArrayList<String> al = new ArrayList<>(); al.add("JDK 1.7"); System.out.println("Array-List enthaelt " + al.size() + " Element"); al = new ArrayList<>(); System.out.println("Array-List enthaelt " + al.size() + " Elemente"); } } Ausgabe Array-List enthaelt 1 Elemente Array-List enthaelt 0 Elemente Aber wie läuft man jetzt über einen Container und gibt z.B. alle Element im Container aus? Für eine ArrayList könnte man dies noch mit einer normalen For-Schleife mit Index-Zähler und dem wahlfreiem Zugriff auf die Array-List (Element-Funktion „get(index)“) umsetzen: import java.util.ArrayList; Hier wird das in Java 5 (JDK 1.5) eingeführte Sprachmittel „Generics“ benutzt. Dieses JavaTutorial behandelt Generics leider nicht. 39 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 159 / 409 public class Kap_09_03_Bsp_03_ContainerSchleife { public static void main(String[] args) { ArrayList<String> al = new ArrayList<>(); al.add("Hallo"); al.add("Java"); al.add("Interessierte"); // Achtung – so laeuft man eigentlich nie ueber einen Container for (int i = 0; i < al.size(); i++) { String s = al.get(i); System.out.print(s + " "); } } } Ausgabe Hallo Java Interessierte Für das normale Laufen über einen Container sollte man nie den wahlfreien Zugriff mit „get(index)“ einsetzen, denn: • Es gibt Container, bei denen der wahlfreie Zugriff sehr langsam ist – z.B. die LinkedList, siehe Kapitel 9.3.5. • Es gibt viele Container, die prinzip-bedingt gar keinen wahlfreien Zugriff unterstützen können, und daher auch nicht enthalten – z.B. eine HashMap, siehe Kapitel 9.3.9. • Außerdem gibt es für einfachere Lösungen für dieses Problem. An die Stelle des wahlfreien Zugriffs treten dann die neue For-Schleife oder die Iteratoren40. Beginnen wir mit der neuen For-Schleife. Hinweis – die Nutzung des wahlfreien Zugriffs sollte nur dann genutzt werden, wenn man diesen explizit durch den Algorithmus benötigt. Dann ist er natürlich sehr sinnvoll. Aber Sie sollten dann auch einen Container wählen, der den wahlfreien Zugriff performant unterstützt, wie z.B. die ArrayList – siehe Kapitel 9.3.4. 9.3.1.1 Neue For-Schleife für Container Mit JDK 1.5 (Java 5) wurde ein neuer For-Schleifentyp eingeführt (siehe auch Kapitel 7.4) – der seit dem die normale Variante ist. Hierbei muß nur noch eine Element-Lauf-Variable mit Typ, und nach einem Doppelpunkt der Container angegeben werden – den Rest macht der Compiler automatisch – siehe Beispiel. import java.util.ArrayList; public class Kap_09_03_Bsp_04_NeueForSchleife { public static void main(String[] args) { ArrayList<String> l = new ArrayList<>(); l.add("Java"); l.add("mit"); l.add("neuer"); l.add("JDK 1.5"); Für eine ausführliche Diskussion von Iteratoren siehe das Buch „Entwurfsmuster“ von u.a. Erich Gamma. 40 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 160 / 409 l.add("For-Schleife"); for (String s : l) { System.out.print(s + " "); } System.out.println(); } } Ausgabe Java mit neuer JDK 1.5 For-Schleife Diese Schleife kann eigentlich nur eins: einfach über den Container laufen – Element für Element. Mehr nicht, aber auch nicht weniger – und das ist der normale Anwendungsfall für Container mit Schleifen, der 95 % Fall. Und den beherrscht sie einfach, schnell und zuverlässig. Und darum nehmen wir diese Schleife auch immer für diesen Use-Case. Die einzigen zusätzlichen Möglichkeiten der Container-For-Schleife sind die Nutzung der Sprung-Anweisungen „break“ und „continue“ – siehe Kapitel 7.4 – die die Schleife vorzeitig verlassen oder direkt zum nächsten Element gehen können. Hier ein Beispiel mit der neuen For-Schleife mit „break“ und „continue“: import java.util.ArrayList; public class Appl { public static void main(String[] args) { ArrayList<String> l = new ArrayList<String>(); l.add("a"); l.add("b"); l.add("c"); l.add("d"); l.add("e"); l.add("f"); for (String s : l) { System.out.print(s); if (s.equals("b")) { continue; } System.out.print("-"); if (s.equals("e")) { break; } } System.out.println(); // Zeile (*) // Zeile (**) } } Ausgabe a-bc-d-e- Nach der Ausgabe von „b “ wird kein Bindestrich ausgegeben, da das „continue“ in Zeile (*) zuschlägt, und nach Ausgabe von „e-“ wird die Schleife aufgrund von „break“ in Zeile (**) abgebrochen. 9.3.1.2 Container & Iteratoren Ein Iterator ist die Abstraktion eines Objekts mit dem über eine Objekt-Menge gelaufen (iteriert) © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 161 / 409 werden kann. Ein Iterator zeigt auf ein Objekt, kann auf das nächste gesetzt werden, und weiß, ob es noch weitere Objekte in der Menge gibt. Jeder Java Container hat eine Element-Funktion „iterator()“, die einen Iterator über den Container zurückgibt. Mit der Element-Funktion „hasNext()“ kann am Iterator abgefragt werden, ob noch weitere Elemente vorliegen. Die Element-Funktion „next()“ geht vor das nächste Element und gibt dabei das aktuelle Element zurück: import java.util.ArrayList; import java.util.Iterator; public class Appl { public static void main(String[] args) { ArrayList<String> al = new ArrayList<>(); al.add("Lasst"); al.add("uns"); al.add("Java"); al.add("lernen"); // Iterator mit While-Schleife Iterator<String> it1 = al.iterator(); while (it1.hasNext()) { String s = it1.next(); System.out.print(s + " "); } System.out.println(); // Iterator mit For-Schleife for (Iterator<String> it2 = al.iterator(); it2.hasNext(); ) { String s = it2.next(); System.out.print(s + " "); } System.out.println(); } } Ausgabe Lasst uns Java lernen Lasst uns Java lernen In der Praxis trifft man Iteratoren sowohl mit While- als auch mit For-Schleife – darum enthält das obige Beispiel beide Varianten. Sie sind gleichwertig und es ist daher reine Geschmackssache, welche Variante Sie bevorzugen. 9.3.2 Untypisierte Container Alle bisherigen Beispiel arbeiten mit typisierten Containern, d.h. Containern, deren Element-Typ mit spitzen Klammern angegeben ist41. Diese Typisierung hat zwei Vorteile: • Sie können nur Elemente passendes Typs in den Container einfügen – Objekte mit falschem Typ erzeugen einen Compiler-Fehler (siehe Beispiel weiter oben). • Holen Sie Element aus dem Container, so ist der Typ bekannt und spezielle Konvertierungen sind nicht notwendig. Hier wird das in Java 5 (JDK 1.5) eingeführte Sprachmittel „Generics“ benutzt. Dieses JavaTutorial behandelt Generics leider nicht. 41 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 162 / 409 In der Praxis begegnet uns noch sehr häufig Code mit untypisierten Containern, da dies über 9 Jahre Stand der Java Technik war: • Viele Schnittstellen sind in dieser Ära entstanden, setzen daher auf untypisierten Container auf, und sind noch heute aktuell. • Auch heute noch viel alter Java Code exisitiert und genutzt wird. Von daher sollten Sie auch die Verwendung von untypisierten Containern kennen – in neuem Code sollten Sie aber nur typisierte Container verwenden. Das folgende Beispiel zeigt wieder die schon bekannte Nutzung der ArrayList, diesmal aber ohne Typisierung: import java.util.ArrayList; import java.util.Iterator; public class Kap_09_03_Bsp_07_UntypisierteContainer { public static void main(String[] args) { ArrayList al = new ArrayList(); al.add("Untypisierter"); al.add("Container"); al.add("-"); al.add("ohne"); al.add("Generics"); Iterator it = al.iterator(); while (it.hasNext()) { String s = (String) it.next(); System.out.print(s + " "); } System.out.println(); // (*) // (**) // (***) } } Ausgabe Untypisierter Container - ohne Generics Bevor wir den obigen Code besprechen, erstmal ein wichtiger Hinweis: der obige Code erzeugt sowohl im Java-Compiler „javac“ auf der Kommando-Zeile, als auch in den heutigen IDE's wie der Eclipse Warnungen. Die beiden folgenden Abbildungen zeigen dies: Abb. 9-3 : Warnungen des Java-Compilers wegen der Nutzung untypisierter Container © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 163 / 409 Abb. 9-4 : Warnungen der Eclipse 3.7.2 wegen der Nutzung untypisierter Container Das freut mich – sowohl der Java-Compiler als auch die Eclipse sind meiner Meinung: Nutzen Sie keine untypisierten Container. Wenn Sie untypisierte Container aber doch nutzen müssen, dann unterdrücken Sie diese Warnungen, damit die interessanten Warnungen nicht verdeckt werden. Dies können Sie z.B. mit der seit Java 5 (JDK 1.5) vorhandenen Annotation „@SuppressWarnings“ machen42. Die Eclipse kann sie automatisch als Quick-Fix für Sie einfügen. Der neue Quelltext sieht dann so aus43: import java.util.ArrayList; import java.util.Iterator; public class Kap_09_03_Bsp_07_UntypisierteContainer { Annotations sind ein weiteres neues Sprachmittel von Java 5 (JDK 1.5). Auch dies wird in diesem Java-Tutorial leider nicht besprochen. 43 So (mit Annotation) finden Sie den Quelltext auch unter den fertigen Code-Beispielen auf meiner Homepage http://www.wilkening-online.de. 42 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 164 / 409 @SuppressWarnings({ "rawtypes", "unchecked" }) public static void main(String[] args) { ArrayList al = new ArrayList(); al.add("Untypisierter"); al.add("Container"); al.add("-"); al.add("ohne"); al.add("Generics"); // (*) Iterator it = al.iterator(); while (it.hasNext()) { String s = (String) it.next(); System.out.print(s + " "); } System.out.println(); // (**) // (***) } } Ausgabe Untypisierter Container - ohne Generics Sowohl der Container als auch der Iterator werden hier ohne Element-Typ angegeben – siehe Zeile (*) und (**). Prinzipiell sieht der Code dadurch einfacher aus, aber er ist nur unsicherer: • Mit „add“ können jetzt beliebige Objekte dem Container hinzugefügt werden – nicht nur Strings. • Und die Iterator-Funktion „next“ (Zeile (***)) gibt das Element jetzt nur als „Object“ (siehe Kapitel 14.11) zurück – und dies muß nun explizit von uns gecastet werden (Angabe des Zieltyps in runden Klammern vor dem Funktions-Aufruf). Enthält der Container andere Elemente, so wird diese Zeile zur Laufzeit schief gehen und eine Class-Cast Exception werfen. Das folgende Beispiel zeigt das typische Problem untypisierter Container. Das Problem ist, dass der Fehler nicht vom Compiler erkannt wird, sondern erst zur Laufzeit auftritt – und wenn Sie Pech haben, erst beim Kunden. import java.util.ArrayList; import java.util.Iterator; public class Kap_09_03_Bsp_08_LaufzeitFehler { @SuppressWarnings({ "rawtypes", "unchecked" }) public static void main(String[] args) { ArrayList al = new ArrayList(); al.add("Untypisierter"); al.add("Container"); al.add("-"); al.add(new StringBuilder()); // Achtung - kein String al.add("Generics"); Iterator it = al.iterator(); while (it.hasNext()) { String s = (String) it.next(); System.out.print(s + " "); } System.out.println(); // Laufzeit-Fehler beim // vierten Durchlauf } } Ausgabe Untypisierter Container – Exception in thread "main" java.lang.ClassCastException: © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 165 / 409 java.lang.StringBuilder cannot be cast to java.lang.String at Kap_09_03_Bsp_08_LaufzeitFehler.main (Kap_09_03_Bsp_08_LaufzeitFehler.java:17) Hinweis – zur Übung werden alle Container in den folgenden Container-Kapiteln sowohl typisiert als auch untypisiert vorgestellt – bevorzugen Sie aber, wann immer möglich, die typisierte Variante. In den restlichen Beispielen und Musterlösungen des Java-Tutorials kommen daher dann auch nur noch die typisierten Varianten vor. 9.3.3 Container Alle Container können hier nicht annähernd vorgestellt werden – dazu gibt zuviele in der Java Bibliothek. Statt dessen beschränken wir uns hier auf die wichtigsten Container: • java.util.ArrayList – Dynamisches Array (Kapitel 9.3.4) • java.util.LinkedList – doppelt verkettete Liste (Kapitel 9.3.5) • java.util.TreeSet – Sortierte Menge (Kapitel 9.3.6) • java.util.HashSet – Unsortierte gehashte Menge (Kapitel 9.3.7) • java.util.TreeMap – Sortierter assoziativer Container (Kapitel 9.3.8) • java.util.HashMap – Unsortierter gehashter assoziativer Container (Kapitel 9.3.9) Alle Container sind intern typlose Container, d.h. sie können alle Arten von Typen, die nicht elementar sind, aufnehmen – d.h. alle Objekte. Dies hat manchmal Vorteile, führt aber in der Praxis häufig zu Typ-Fehlern, die erst zur Laufzeit auffallen (vergleiche vorhergehendes Kapitel 9.3.2). Daher nutzen Sie bitte immer die typisierte Variante. In den folgenden Kapiteln werden die Container der Vollständigkeit halber aber sowohl typisiert als auch untypisiert vorgestellt44. Hinweis – um die Klassen ArrayList, TreeMap, Iterator, usw. benutzen zu können, müssen die entsprechenden „import“ Anweisungen am Anfang des Quelltextes (nach der packageAnweisung und vor der Klassen-Definition) stehen – siehe Beispiele. Weitere Informationen zu Packages und Imports finden Sie im Kapitel über Packages – siehe Kapitel 13. Achtung – alle Java Container können nur Objekte aufnehmen, und keine elementaren Datentypen. Einfache Bool-, Zeichen-, Integer- oder Fließkomma-Werte können Sie also nicht direkt in Java Containern speichern. Sie müssen solche Werte in spezielle Wrapper-Klassen einschliessen – siehe Kapitel 9.4. Seit dem JDK 1.5 geschieht dieses Wrappen automatisch, dieses Java Feature nennt sich „Auto-Boxing“ – siehe Kapitel 9.4.1. Lassen Sie sich aber nicht täuschen – Java Container können weiterhin keine elementaren Datentypen aufnehmen. Der Effekt hat nichts an der internen Vorgehensweise und Implementierung geändert. 9.3.4 ArrayList Die ArrayList ein dynamisches Array, d.h. ein Container bei dem die Elemente direkt Während das Java-Tutorial genügend Raum für die typisierten Container hat, wird das Sprachmittel der Generics im Java-Tutorial leider nicht eingeführt. 44 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 166 / 409 hintereinander im Speicher liegen und problemlos am Ende angefügt werden können. Ausserdem ist er ein Beispiel für einen sequentiellen Container. Das ist ein Container, in dem die Objekte sequentiell hintereinander liegen (z.B. in der Reihenfolge des Einfügens) und der Container hierbei keinerlei Einfluss auf die Reihenfolge der Objekte nimmt. todo import java.util.ArrayList; import java.util.Iterator; public class Appl { public static void main(String[] args) { ArrayList<String> al = new ArrayList<>(); al.add("Hallo"); al.add("wunderbarer"); al.add("Java"); al.add("Kurs"); System.out.println("Der Container enthaelt " + al.size() + " Objekte"); // Neue JDK 1.5 For-Schleife for (String s : al) { System.out.print(s + " "); } System.out.println(); // Iterator-Loesung mit For-Schleife for (Iterator<String> it = al.iterator(); it.hasNext();) { String s = it.next(); System.out.print(s + " "); } System.out.println(); // Iterator-Loesung mit While-Schleife Iterator<String> it = al.iterator(); while (it.hasNext()) { String s = it.next(); System.out.print(s + " "); } System.out.println(); } } Ausgabe Der Container enthaelt Hallo wunderbarer Java Hallo wunderbarer Java Hallo wunderbarer Java 4 Objekte Kurs Kurs Kurs import java.util.ArrayList; import java.util.Iterator; public class Appl { @SuppressWarnings({ "unchecked", "rawtypes" }) public static void main(String[] args) { ArrayList al = new ArrayList(); al.add("Hallo"); al.add("wunderbarer"); al.add("Java"); al.add("Kurs"); System.out.println("Der Container enthaelt " + al.size() + " Objekte"); // Neue JDK 1.5 For-Schleife © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 167 / 409 for (Object o : al) { String s = (String) o; System.out.print(s + " "); } System.out.println(); // Iterator-Loesung mit For-Schleife for (Iterator it = al.iterator(); it.hasNext();) { String s = (String) it.next(); System.out.print(s + " "); } System.out.println(); // Iterator-Loesung mit While-Schleife Iterator it = al.iterator(); while (it.hasNext()) { String s = (String) it.next(); System.out.print(s + " "); } System.out.println(); } } Ausgabe Der Container enthaelt Hallo wunderbarer Java Hallo wunderbarer Java Hallo wunderbarer Java 4 Objekte Kurs Kurs Kurs 9.3.5 LinkedList todo import java.util.Iterator; import java.util.LinkedList; public class Appl { public static void main(String[] args) { LinkedList<String> ll = new LinkedList<>(); ll.add("Hallo"); ll.add("wunderbarer"); ll.add("Java"); ll.add("Kurs"); System.out.println("Der Container enthaelt " + ll.size() + " Objekte"); // Neue JDK 1.5 For-Schleife for (String s : ll) { System.out.print(s + " "); } System.out.println(); // Iterator-Loesung mit For-Schleife for (Iterator<String> it = ll.iterator(); it.hasNext();) { String s = it.next(); System.out.print(s + " "); } System.out.println(); // Iterator-Loesung mit While-Schleife Iterator<String> it = ll.iterator(); while (it.hasNext()) { String s = it.next(); System.out.print(s + " "); } System.out.println(); } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 168 / 409 } Ausgabe Der Container enthaelt Hallo wunderbarer Java Hallo wunderbarer Java Hallo wunderbarer Java 4 Objekte Kurs Kurs Kurs import java.util.Iterator; import java.util.LinkedList; public class Appl { @SuppressWarnings({ "unchecked", "rawtypes" }) public static void main(String[] args) { LinkedList ll = new LinkedList(); ll.add("Hallo"); ll.add("wunderbarer"); ll.add("Java"); ll.add("Kurs"); System.out.println("Der Container enthaelt " + ll.size() + " Objekte"); // Neue JDK 1.5 For-Schleife for (Object o : ll) { String s = (String) o; System.out.print(s + " "); } System.out.println(); // Iterator-Loesung mit For-Schleife for (Iterator it = ll.iterator(); it.hasNext();) { String s = (String) it.next(); System.out.print(s + " "); } System.out.println(); // Iterator-Loesung mit While-Schleife Iterator it = ll.iterator(); while (it.hasNext()) { String s = (String) it.next(); System.out.print(s + " "); } System.out.println(); } } Ausgabe Der Container enthaelt Hallo wunderbarer Java Hallo wunderbarer Java Hallo wunderbarer Java 4 Objekte Kurs Kurs Kurs 9.3.6 TreeSet todo import java.util.Iterator; import java.util.TreeSet; public class Appl { public static void main(String[] args) { TreeSet<String> ts = new TreeSet< >(); ts.add("Detlef"); ts.add("Edgar"); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 169 / 409 ts.add("Silke"); ts.add("Martina"); ts.add("Aaron"); System.out.println("Der Container enthaelt " + ts.size() + " Objekte"); boolean exists = ts.contains("Detlef"); System.out.println("Detlef ist im TreeSet vorhanden: " + exists); exists = ts.contains("Hans"); System.out.println("Hans ist im TreeSet vorhanden: " + exists); // Neue JDK 1.5 For-Schleife for (String s : ts) { System.out.print(s + " "); } System.out.println(); // Iterator-Loesung mit For-Schleife for (Iterator<String> it = ts.iterator(); it.hasNext();) { String s = it.next(); System.out.print(s + " "); } System.out.println(); // Iterator-Loesung mit While-Schleife Iterator<String> it = ts.iterator(); while (it.hasNext()) { String s = it.next(); System.out.print(s + " "); } System.out.println(); } } Ausgabe Der Container enthaelt 5 Objekte Detlef ist im TreeSet vorhanden: true Hans ist im TreeSet vorhanden: false Aaron Detlef Edgar Martina Silke Aaron Detlef Edgar Martina Silke Aaron Detlef Edgar Martina Silke import java.util.Iterator; import java.util.TreeSet; public class Appl { @SuppressWarnings({ "unchecked", "rawtypes" }) public static void main(String[] args) { TreeSet ts = new TreeSet(); ts.add("Detlef"); ts.add("Edgar"); ts.add("Silke"); ts.add("Martina"); ts.add("Aaron"); System.out.println("Der Container enthaelt " + ts.size() + " Objekte"); boolean exists = ts.contains("Detlef"); System.out.println("Detlef ist im TreeSet vorhanden: " + exists); exists = ts.contains("Hans"); System.out.println("Hans ist im TreeSet vorhanden: " + exists); // Neue JDK 1.5 For-Schleife for (Object o : ts) { String s = (String) o; System.out.print(s + " "); } System.out.println(); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 170 / 409 // Iterator-Loesung mit For-Schleife for (Iterator it = ts.iterator(); it.hasNext();) { String s = (String) it.next(); System.out.print(s + " "); } System.out.println(); // Iterator-Loesung mit While-Schleife Iterator it = ts.iterator(); while (it.hasNext()) { String s = (String) it.next(); System.out.print(s + " "); } System.out.println(); } } Ausgabe Der Container enthaelt 5 Objekte Detlef ist im TreeSet vorhanden: true Hans ist im TreeSet vorhanden: false Aaron Detlef Edgar Martina Silke Aaron Detlef Edgar Martina Silke Aaron Detlef Edgar Martina Silke 9.3.7 HashSet todo import java.util.HashSet; import java.util.Iterator; public class Appl { public static void main(String[] args) { HashSet<String> hs = new HashSet<>(); hs.add("Detlef"); hs.add("Edgar"); hs.add("Silke"); hs.add("Martina"); hs.add("Aaron"); System.out.println("Der Container enthaelt " + hs.size() + " Objekte"); boolean exists = hs.contains("Detlef"); System.out.println("Detlef ist im HashSet vorhanden: " + exists); exists = hs.contains("Hans"); System.out.println("Hans ist im HashSet vorhanden: " + exists); // Neue JDK 1.5 For-Schleife for (String s : hs) { System.out.print(s + " "); } System.out.println(); // Iterator-Loesung mit For-Schleife for (Iterator<String> it = hs.iterator(); it.hasNext();) { String s = it.next(); System.out.print(s + " "); } System.out.println(); // Iterator-Loesung mit While-Schleife Iterator<String> it = hs.iterator(); while (it.hasNext()) { String s = it.next(); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 171 / 409 System.out.print(s + " "); } System.out.println(); } } Mögliche Ausgabe (Die Reihenfolge im Hash-Container ist undefiniert) Der Container enthaelt 5 Objekte Detlef ist im HashSet vorhanden: true Hans ist im HashSet vorhanden: false Aaron Martina Edgar Silke Detlef Aaron Martina Edgar Silke Detlef Aaron Martina Edgar Silke Detlef import java.util.HashSet; import java.util.Iterator; public class Appl { @SuppressWarnings({ "unchecked", "rawtypes" }) public static void main(String[] args) { HashSet hs = new HashSet(); hs.add("Detlef"); hs.add("Edgar"); hs.add("Silke"); hs.add("Martina"); hs.add("Aaron"); System.out.println("Der Container enthaelt " + hs.size() + " Objekte"); boolean exists = hs.contains("Detlef"); System.out.println("Detlef ist im HashSet vorhanden: " + exists); exists = hs.contains("Hans"); System.out.println("Hans ist im HashSet vorhanden: " + exists); // Neue JDK 1.5 For-Schleife for (Object o : hs) { String s = (String) o; System.out.print(s + " "); } System.out.println(); // Iterator-Loesung mit For-Schleife for (Iterator it = hs.iterator(); it.hasNext();) { String s = (String) it.next(); System.out.print(s + " "); } System.out.println(); // Iterator-Loesung mit While-Schleife Iterator it = hs.iterator(); while (it.hasNext()) { String s = (String) it.next(); System.out.print(s + " "); } System.out.println(); } } Mögliche Ausgabe (Die Reihenfolge im Hash-Container ist undefiniert) Der Container enthaelt 5 Objekte Detlef ist im HashSet vorhanden: true Hans ist im HashSet vorhanden: false Aaron Martina Edgar Silke Detlef Aaron Martina Edgar Silke Detlef Aaron Martina Edgar Silke Detlef © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 172 / 409 9.3.8 TreeMap todo Bei einem assoziativen Container werden Schlüssel/Wert Paare gespeichert, d.h. dass eigentliche Objekt wird über einen Schlüssel referenziert. Ein Beispiel dafür ist der ausgewogene binäre rot-schwarz Baum „TreeMap“, der außerdem noch implizit die Werte nach dem Schlüssel sortiert. import java.util.Iterator; import java.util.Map.Entry; import java.util.TreeMap; public class Appl { public static void main(String[] args) { TreeMap<String, String> tm = new TreeMap<>(); tm.put("Detlef", "1234"); tm.put("Edgar", "5678"); tm.put("Silke", "248"); tm.put("Martina", "999"); tm.put("Aaron", "646"); System.out.println("Der Container enthaelt " + tm.size() + " Objekte"); String s = tm.get("Edgar"); if (s != null) System.out.println("Edgar hat die Nummer: " + s); else System.out.println("Edgar ist nicht in der TreeMap"); s = tm.get("Hans"); if (s != null) System.out.println("Hans hat die Nummer: " + s); else System.out.println("Hans ist nicht in der TreeMap"); // Neue JDK 1.5 For-Schleife for (Entry<String, String> e : tm.entrySet()) { String key = e.getKey(); String val = e.getValue(); System.out.print(key + " => " + val + ", "); } System.out.println(); // Iterator-Loesung mit For-Schleife for (Iterator<Entry<String, String>> it = tm.entrySet().iterator(); it.hasNext();) { Entry<String, String> e = it.next(); String key = e.getKey(); String val = e.getValue(); System.out.print(key + " => " + val + ", "); } System.out.println(); // Iterator-Loesung mit While-Schleife Iterator<Entry<String, String>> it = tm.entrySet().iterator(); while (it.hasNext()) { Entry<String, String> e = it.next(); String key = e.getKey(); String val = e.getValue(); System.out.print(key + " => " + val + ", "); } System.out.println(); } } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 173 / 409 Ausgabe Der Container enthaelt 5 Objekte Edgar hat die Nummer: 5678 Hans ist nicht in der TreeMap Aaron => 646, Detlef => 1234, Edgar => 5678, Martina => 999, Silke => 248, Aaron => 646, Detlef => 1234, Edgar => 5678, Martina => 999, Silke => 248, Aaron => 646, Detlef => 1234, Edgar => 5678, Martina => 999, Silke => 248, import java.util.Iterator; import java.util.Map.Entry; import java.util.TreeMap; public class Appl { @SuppressWarnings({ "unchecked", "rawtypes" }) public static void main(String[] args) { TreeMap tm = new TreeMap(); tm.put("Detlef", "1234"); tm.put("Edgar", "5678"); tm.put("Silke", "248"); tm.put("Martina", "999"); tm.put("Aaron", "646"); System.out.println("Der Container enthaelt " + tm.size() + " Objekte"); String s = (String) tm.get("Edgar"); if (s != null) System.out.println("Edgar hat die Nummer: " + s); else System.out.println("Edgar ist nicht in der TreeMap"); s = (String) tm.get("Hans"); if (s != null) System.out.println("Hans hat die Nummer: " + s); else System.out.println("Hans ist nicht in der TreeMap"); // Neue JDK 1.5 For-Schleife for (Object o : tm.entrySet()) { Entry<String, String> e = (Entry<String, String>) o; String key = e.getKey(); String val = e.getValue(); System.out.print(key + " => " + val + ", "); } System.out.println(); // Iterator-Loesung mit For-Schleife for (Iterator it = tm.entrySet().iterator(); it.hasNext();) { Entry<String, String> e = (Entry<String, String>) it.next(); String key = e.getKey(); String val = e.getValue(); System.out.print(key + " => " + val + ", "); } System.out.println(); // Iterator-Loesung mit While-Schleife Iterator<Entry<String, String>> it = tm.entrySet().iterator(); while (it.hasNext()) { Entry<String, String> e = it.next(); String key = e.getKey(); String val = e.getValue(); System.out.print(key + " => " + val + ", "); } System.out.println(); } } Ausgabe Der Container enthaelt 5 Objekte Edgar hat die Nummer: 5678 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Hans ist Aaron => Aaron => Aaron => Seite 174 / 409 nicht in der TreeMap 646, Detlef => 1234, Edgar => 5678, Martina => 999, Silke => 248, 646, Detlef => 1234, Edgar => 5678, Martina => 999, Silke => 248, 646, Detlef => 1234, Edgar => 5678, Martina => 999, Silke => 248, 9.3.9 HashMap todo import java.util.HashMap; import java.util.Iterator; import java.util.Map.Entry; public class Appl { public static void main(String[] args) { HashMap<String, String> hm = new HashMap<>(); hm.put("Detlef", "1234"); hm.put("Edgar", "5678"); hm.put("Silke", "248"); hm.put("Martina", "999"); hm.put("Aaron", "646"); System.out.println("Der Container enthaelt " + hm.size() + " Objekte"); String s = hm.get("Edgar"); if (s != null) System.out.println("Edgar hat die Nummer: " + s); else System.out.println("Edgar ist nicht in der HashMap"); s = hm.get("Hans"); if (s != null) System.out.println("Hans hat die Nummer: " + s); else System.out.println("Hans ist nicht in der HashMap"); // Neue JDK 1.5 For-Schleife for (Entry<String, String> e : hm.entrySet()) { String key = e.getKey(); String val = e.getValue(); System.out.print(key + " => " + val + ", "); } System.out.println(); // Iterator-Loesung mit For-Schleife for (Iterator<Entry<String, String>> it = hm.entrySet().iterator(); it.hasNext();) { Entry<String, String> e = it.next(); String key = e.getKey(); String val = e.getValue(); System.out.print(key + " => " + val + ", "); } System.out.println(); // Iterator-Loesung mit While-Schleife Iterator<Entry<String, String>> it = hm.entrySet().iterator(); while (it.hasNext()) { Entry<String, String> e = it.next(); String key = e.getKey(); String val = e.getValue(); System.out.print(key + " => " + val + ", "); } System.out.println(); } } Mögliche Ausgabe (Die Reihenfolge im Hash-Container ist undefiniert) © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 175 / 409 Der Container enthaelt 5 Objekte Edgar hat die Nummer: 5678 Hans ist nicht in der HashMap Aaron => 646, Martina => 999, Edgar => 5678, Silke => 248, Detlef => 1234, Aaron => 646, Martina => 999, Edgar => 5678, Silke => 248, Detlef => 1234, Aaron => 646, Martina => 999, Edgar => 5678, Silke => 248, Detlef => 1234, import java.util.HashMap; import java.util.Iterator; import java.util.Map.Entry; public class Appl { @SuppressWarnings({ "unchecked", "rawtypes" }) public static void main(String[] args) { HashMap hm = new HashMap(); hm.put("Detlef", "1234"); hm.put("Edgar", "5678"); hm.put("Silke", "248"); hm.put("Martina", "999"); hm.put("Aaron", "646"); System.out.println("Der Container enthaelt " + hm.size() + " Objekte"); String s = (String) hm.get("Edgar"); if (s != null) System.out.println("Edgar hat die Nummer: " + s); else System.out.println("Edgar ist nicht in der HashMap"); s = (String) hm.get("Hans"); if (s != null) System.out.println("Hans hat die Nummer: " + s); else System.out.println("Hans ist nicht in der HashMap"); // Neue JDK 1.5 For-Schleife for (Object o : hm.entrySet()) { Entry<String, String> e = (Entry<String, String>) o; String key = e.getKey(); String val = e.getValue(); System.out.print(key + " => " + val + ", "); } System.out.println(); // Iterator-Loesung mit For-Schleife for (Iterator it = hm.entrySet().iterator(); it.hasNext();) { Entry<String, String> e = (Entry<String, String>) it.next(); String key = e.getKey(); String val = e.getValue(); System.out.print(key + " => " + val + ", "); } System.out.println(); // Iterator-Loesung mit While-Schleife Iterator it = hm.entrySet().iterator(); while (it.hasNext()) { Entry<String, String> e = (Entry<String, String>) it.next(); String key = e.getKey(); String val = e.getValue(); System.out.print(key + " => " + val + ", "); } System.out.println(); } } Mögliche Ausgabe (Die Reihenfolge im Hash-Container ist undefiniert) Der Container enthaelt 5 Objekte Edgar hat die Nummer: 5678 Hans ist nicht in der HashMap © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 176 / 409 Aaron => 646, Martina => 999, Edgar => 5678, Silke => 248, Detlef => 1234, Aaron => 646, Martina => 999, Edgar => 5678, Silke => 248, Detlef => 1234, Aaron => 646, Martina => 999, Edgar => 5678, Silke => 248, Detlef => 1234, 9.3.10 Container-Vergleich todo 9.4 Wrapper Klassen In Java gibt es für alle elementaren Datentypen (inkl. „void“) sogenannte Wrapper-Klassen. Sie haben drei Aufgaben: • Sie dienen als Wrapper-Klassen für z.B. das Container-Framework – siehe Kapitel 9.3.3 und das folgende Kapitel 9.4.1. Hinweis – die Wrapper-Objekte sind wie Strings unveränderbar („immutable“), d.h. die Werte in den Objekten lassen sich nicht ändern. • Sie sammeln zu dem Typ gehörige allgemeine Funktionen in Form von Klassen-Funktionen. Z.B. die Funktionen zum Konvertieren, wie „parseInt“ – siehe Kapitel 3.11. • Sie werden bei Reflexion als Meta-Klassen eingesetzt – siehe Kapitel 24. Dies erklärt auch die Wrapper-Klasse „Void“, die sonst keinen Sinn machen würde. Elementarer Datentyp boolean char byte short int long float double void Wrapper-Klasse Boolean Character Byte Short Integer Long Float Double Void Bis auf die Wrapper-Klassen für „char“ und „int“ haben sie den gleichen Namen wie der jeweilige elementare Datentyp den sie kapseln, beginnen aber groß45. 9.4.1 Auto-Boxing In Kapitel 9.3.3 habe ich geschrieben, dass die Container-Klassen keine elementaren DatenTypen aufnehmen können, sondern nur Objekte. Das läßt uns keine Ruhe, und darum stellen wir diese Aussage hier auf die Probe: import java.util.ArrayList; import java.util.Iterator; public class Kap_09_04_Bsp_01_AutoBoxing { Warum heissen die Wrapper Klassen für „char“ und „int“ nicht wie die elementaren Datentypen, sondern anders? Ich weiß es nicht, vermute aber historische Gründe. 45 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 177 / 409 @SuppressWarnings({ "unchecked", "rawtypes" }) public static void main(String[] args) { ArrayList al = new ArrayList(); al.add(1); // "1" ist ein "int" al.add(6); // "6" ist ein "int" al.add(23); // "23" ist ein "int" al.add(42); // "42" ist ein "int" System.out.println("Die Liste enthaelt " + al.size() + " ints"); Iterator it = al.iterator(); while (it.hasNext()) { int n = (int) it.next(); System.out.print(n + " "); } // Cast in "int" } } Ausgabe Die Liste enthaelt 4 ints 1 6 23 42 Funktioniert doch! Was erzählt der denn da? Nun, wenn Sie Kapitel 9.3.3 vollständig in Ruhe lesen, dann finden Sie noch eine weitere Aussage: „Elementare Datentypen müssen in speziellen Wrapper-Klassen (siehe Kapitel 9.4) gewrappt werden. Seit dem JDK 1.5 kann dieses Wrappen implizit geschehen (Auto-Boxing) – dies hat aber nichts an der internen Vorgehensweise und Implementierung geändert“. Hinweis – das Beispiel arbeitet mit einem untypisierten Container, da eine Typisierung auf „int“ zu einem Compiler-Fehler geführt hätte, und wir mit der korrekten Typisierung auf „Integer“ die „Überraschung“ viel kleiner gewesen wäre. Intern passiert also das im folgenden Beispiel explizit programmierte – nur das Sie es seit dem JDK 1.5 nicht sehen, da der Compiler es automatisch für Sie macht. Aber Sie dürfen es natürlich immer noch selber machen, bzw. bei einem älteren JDK (vor 1.5) müssen Sie es sogar. import java.util.ArrayList; import java.util.Iterator; public class Appl { public static void main(String[] args) { ArrayList<Integer> al = new ArrayList<>(); al.add(Integer.valueOf(1)); al.add(Integer.valueOf(6)); al.add(Integer.valueOf(23)); al.add(Integer.valueOf(42)); System.out.println("Die Liste enthaelt " + al.size() + " Integer"); Iterator<Integer> it = al.iterator(); while (it.hasNext()) { Integer i = (Integer) it.next(); int n = i.intValue(); System.out.print(n + " "); } } } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 178 / 409 Ausgabe Die Liste enthaelt 4 ints 1 6 23 42 • Beim Einfügen in den Container wird der elementare Datentyp (hier „int“) in eine WrapperKlasse (hier „Integer“) gewrappt („geboxt“). Beim Auto-Boxing geschieht dies automatisch. • Möglicherweise haben Sie beim Einfügen „new Integer(x)“ statt „Integer.valueOf(x)“ erwartet – immerhin müssen alle Objekte mit „new“ erzeugt werden. Im Prinzip würde „new Integer(x)“ hier auch funktionieren, aber die Klassen-Funktion „valueOf“ gibt auch ein entsprechendes Java Integer-Objekt zurück – kann aber schneller sein. Für genaue Details schauen Sie bitte in die Java Referenz-Dokumentation. • Beim Wandeln des Integer-Objekts in einen „int“ muß die Element-Funktion „intValue" genutzt werden. Beim Auto-Boxing geschieht auch dies automatisch. • Die ArrayList ist in diesem Beispiel auf „Integer“ typisiert – das ist die passende Typisierung um via Auto-Boxing „int“s aufzunehmen. Intern bleibt es aber so, dass Java Container keine elementaren Datentypen aufnehmen können. Aber seit dem JDK 1.5 hilft uns die Sprache hier ganz automatisch. 9.4.2 Hierarchie Alle numerischen Wrapper Klassen sind von der Klasse „java.lang.Number“ abgeleitet. Abb. 9-5 : Vererbungs-Hierarchie der Wrapper-Klassen 9.5 Zufalls-Zahlen Um Zufalls-Zahlen zu erzeugen gibt es im Package „java.util“ die Klasse „Random“. Man erzeugt sich ein Objekt der Klasse „Random“ und kann sich von diesem Objekt Zufalls-Zahlen erzeugen lassen. Es existieren Element-Funktionen zur Erzeugung von Zufalls-Zahlen für die wichtigsten Typen („boolean“, „int“, „long“, „float“ und „double“) und einige weitere Anwendungsfälle wie z.B. Byte-Arrays. Die einfachste Element-Funktion ist „nextInt“, die einen Int-Paramter bekommt und dann Zufalls-Zahlen zwischen „0“ (inkl.) und dem übergebenden Argument (exkl.) erzeugt. Im folgenden Beispiel also Zahlen von „0-9“. import java.util.Random; public class Kap_09_05_Bsp_01_ZufallsZahlen { public static void main(String[] args) { Random rnd = new Random(); for (int i=0; i<20; i++) { System.out.print(rnd.nextInt(10) + " "); } } } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 179 / 409 mögliche Ausgabe 2 2 5 3 7 0 3 6 4 5 0 7 4 0 3 8 8 7 6 8 Hinweis – vergessen Sie nicht den notwendigen Import von „java.util.Random“. Benötigen Sie Zufalls-Zahlen in einem anderen Bereich (d.h. nicht von „0“ (inkl.) und dem übergebenden Argument (exkl.)), dann verschieben Sie die Zahlen einfach durch eine Addition oder Subtraktion. Um z.B. einen Würfel mit den Zahlen von „1-6“ zu simulieren, benötigen Sie Zahlen von „0-5“ die Sie durch eine Addition mit „1“ in den gewünschten Bereich verschieben. import java.util.Random; public class Kap_09_05_Bsp_02_Wuerfel { public static void main(String[] args) { Random rnd = new Random(); for (int i = 0; i < 10; i++) { System.out.print(rnd.nextInt(6) + 1 + " "); } } } mögliche Ausgabe 3 4 1 6 5 3 3 2 6 2 Hinweise: • Benötigt man reproduzierbare Zufalls-Zahlen, so kann man ein Random-Objekt auch mit einem Seed-Parameter konstruieren, oder bei einem vorhandenen Random-Objekt den Seed mit der Funktion „setSeed“ setzen. • Double-Zufalls-Zahlen kann man sich auch mit der Klassen-Funktion „random“ der Klasse „java.lang.Math“ erzeugen lassen. Die genauen Unterschiede zwischen dieser Funktion und „nextDouble“ aus „Random“ finden Sie in der Java Referenz-Dokumentation. 9.6 Datum und Uhrzeit Für Datum und Uhrzeit Funktionen stehen u.a. folgende Klassen zur Verfügung: • java.util.Date • java.util.Calendar • java.text.DateFormat • java.text.SimpleDateFormat Wird einfach ein Date Objekt erzeugt ("new Date()“), so enthält dieses Objekt das aktuelle Datum und die aktuelle Zeit. Die Klassen „DateFormat“ und „SimpleDateFormat“ bieten Methoden zur Formatierung der Wandlung des Datums in einen String. import java.util.Date; import java.text.DateFormat; import java.text.SimpleDateFormat; public class Kap_09_06_Bsp_01_Datum { public static void main(String[] args) { Date now = new Date(); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 180 / 409 DateFormat df = DateFormat.getDateInstance(); System.out.println(df.format(now)); df = DateFormat.getDateInstance(DateFormat.FULL); System.out.println(df.format(now)); df = DateFormat.getTimeInstance(); System.out.println(df.format(now)); df = DateFormat.getTimeInstance(DateFormat.FULL); System.out.println(df.format(now)); df = DateFormat.getDateTimeInstance(DateFormat.FULL, DateFormat.FULL); System.out.println(df.format(now)); df = new SimpleDateFormat("d.M.yyyy"); System.out.println(df.format(now)); } } mögliche Ausgabe 06.05.2012 Sonntag, 6. Mai 2012 01:05:39 01:05 Uhr MESZ Sonntag, 6. Mai 2012 01:05 Uhr MESZ 6.5.2012 • Die Definition eines Ausgabe-Formats mit „DateFormat“ arbeitet defaultmäßig mit der aktuellen Länderkennung - dies läßt sich natürlich ändern. • „DateFormat“ kann auch Strings parsen – Element-Funktion „parse“. • Mit der Klasse „Calendar“ kann Datums- und Uhrzeit-Arithmetik mit Date-Objekten durchgeführt werden, und es können viele weitere Informationen gewonnen werden. • Achtung – viele Funktion in „Date“ sind „deprecated“ und durch Funktionen der Klasse „Calendar“ ersetzt worden46. Hinweis – mit dem JDK 1.8 ist eine neue Date-Time Bibliothek in Java eingeführt worden. Diese ist viel durchdachter und leistungsfähiger als die alte hier kurz angesprochene Bibliothek. In realen Projekten sollten Sie besser diese nutzen. 9.7 Datei- und Verzeichnis-Handling Die Klasse „java.io.File“ repräsentiert eine Datei oder ein Verzeichnis, und sie bietet viele Funktionen um Dateien und Verzeichnisse zu manipulieren oder Informationen über sie zu erlangen. Hier ein kleiner Auszug aus der Schnittstelle der Klasse „File“: File(String fullName) Erzeugt ein File Objekt für das Element des Datei-Systems mit dem entsprechenden Namen. Der Name ist vollständig inkl. Pfad. Deprecated Klassen oder Funktionen sind veraltet bzw. weisen Probleme auf, und sollten daher nicht mehr benutzt werden. Normalerweise bietet Java in diesen Fällen neue und bessere Elemente an. 46 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 File(String path, String name) Seite 181 / 409 Erzeugt ein File Objekt für das Element des Datei-Systems mit dem entsprechenden Namen. Der Name wird hierbei getrennt in Pfad und eigentlichem Namen übergeben. Der Vorteil ist hier, dass der Benutzer sich nicht um den Trenner im String kümmern muß, der ja plattform-spezifisch ist. boolean exists() Gibt zurück, ob die Datei oder das Verzeichnis existiert. String getAbsolutePath() Gibt den Namen (inkl. vollständigem absolutem Pfad) des Verzeichnisses oder der Datei zurück. boolean isDirectory() Gibt zurück, ob das File-Objekt ein Verzeichnis referenziert. boolean isFile() Gibt zurück, ob das File-Objekt eine Datei referenziert. long length() Gibt bei einer Datei die Größe der referenzierten Datei zurück. String[ ] list() Gibt bei einem Verzeichnis die Namen der Elemente im Verzeichnis (Verzeichnisse und Dateien) als Array von Strings zurück. Als erstes Beispiel ein kleines Programm, dass überprüft ob der auf der Kommandozeile übergebene Name im Datei-System existiert, und ob es eine Datei oder ein Verzeichnis ist. Wenn es eine Datei ist, so wird noch die Größe der Datei ausgegeben. import java.io.File; public class Kap_09_07_Bsp_01_File { public static void main(String[] args) { if (args.length!=1) { System.out.println("Fehler - es wird ein Datei-Name erwartet"); return; } String name = args[0]; File file = new File(name); if (!file.exists()) { System.out.println("Element \"" + name + "\" existiert nicht"); return; } if (file.isFile()) { String len = "" + file.length(); System.out.println("\"" + name + "\" hat " + len + " Bytes"); return; } if (file.isDirectory()) { System.out.println("\"" + name + "\" ist ein Verzeichnis"); return; } System.out.println("Element \"" + name + "\" -> unbekannten Typ"); } } Mögliche Ausgabe >java Appl Fehler - es wird ein Datei-Name erwartet © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 182 / 409 Mögliche Ausgabe >java Appl java Element "java" existiert nicht Mögliche Ausgabe >java Appl Kap_09_07_Bsp_01_File.class "Kap_09_07_Bsp_01_File.class" hat 1414 Bytes Mögliche Ausgabe >java Appl . "." ist ein Verzeichnis Hinweise • Für die Verwendung von plattform-unabhängigen Datei-Namen wird in Java die URI Konvention gewählt, für die es eine eigene Klasse „java.net.URI“ gibt. • Lesen und Schreiben von Dateien geschieht in Java mit Streams, die mit einem File-Objekt verbunden sind – siehe Kapitel 23. 9.7.1 Rekursive Datei-Suche Ein typisches Problem, dass wir nun lösen können, ist die tiefe Suche im Dateisystem z.B. nach einer Datei mit einem bestimmten Namen. Bevor Sie sich an dieses Unternehmen wagen, empfehle ich Ihnen, sich sinnvolle Testdaten zu erzeugen – sprich eine kleine Beispiel Verzeichnis-Struktur. Ich habe es leider erlebt, dass Anfänger ihr Programm zum Test auf ihr Root-Dateisystem wie z.B. „C:\“ losgelassen haben, und sich dann gewundert haben dass ihr Programm ewig läuft, Fehler meldet oder scheinbar einfach nur den Rechner lahm legt. Ist doch auch kein Wunder, oder? Bei unseren heutigen Plattengrößen gibt es da ziemlich viele Verzeichnisse zu durchsuchen und Datei-Namen zu vergleichen, dass der Rechner gut belastet ist. Und mit ziemlicher Sicherheit trifft das Programm unterwegs auf Verzeichnisse, für die es keine Rechte hat – und schon regnet es Exceptions, und damit können wir noch gar nicht gut umgehen (das lernen wir ja erst in Kapitel 22). Lange Rede, kurzer Sinn – machen Sie sich eine kleine Test-Daten-Struktur auf Ihrer Platte. Ich habe das gemacht, unter „C:\JavaDateiSystemBeispiele“ – und so sieht sie bei mir aus. Suchen wollen wir später die drei Dateien mit dem Namen „find.txt“. Test-Verzeichnis-Struktur C:\ └ JavaDateiSystemBeispiele └ verzeichnis1 │ └ verzeichnis3 │ │ └ verzeichnis4 │ │ │ └ find.txt │ │ │ └ search.dat │ │ │ └ test.txt │ │ └ test.txt │ └ verzeichnis5 │ │ └ find.txt │ │ └ test.txt │ └ daten.dat │ └ test.txt └ verzeichnis2 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 183 / 409 │ └ find.txt │ └ test.txt └ search.dat └ test.txt Hinweis – bitte ignorieren Sie im Augenblick die blau markierten Dateien „search.dat“ und „daten.dat“ – wir benötigen Sie später in den Aufgaben 9.8.7 und 9.8.8. Damit können wir nun loslegen. Der erste Schritt ist einfach. Wir laufen einfach über das SuchVerzeichnis und geben alles aus, was wir finden – getrennt nach Verzeichnissen und Dateien. Einzige Besonderheit: wir geben den Namen vollständig aus, und setzen ihn nicht selber zusammen, sondern lassen ihn uns vom File-Objekt geben – Zeile (*). Hintergrund hierfür sind die unterschiedlichen Datei-Trenner unter z.B. Windows (Backslash „\“) und Linux (Slash „/“), um die wir uns sonst selbst kümmern müßten. import java.io.File; public class Kap_09_07_Bsp_02_FileSucheRekursivStep1 { public static void main(String[] args) { File path = new File("C:\\JavaDateiSystemBeispiele"); String[] filenames = path.list(); for (String filename : filenames) { File file = new File(path, filename); String fullfilename = file.getAbsolutePath(); if (file.isDirectory()) { System.out.println("Verz.: " + fullfilename); } else if (file.isFile()) { System.out.println("Datei: " + fullfilename); } } } // (*) } Mögliche Ausgabe (andere Reihenfolge möglich – ist nicht definiert) Datei: C:\JavaDateiSystemBeispiele\search.dat Datei: C:\JavaDateiSystemBeispiele\test.txt Verz.: C:\JavaDateiSystemBeispiele\verzeichnis1 Verz.: C:\JavaDateiSystemBeispiele\verzeichnis2 Der nächste Schritt ist minimal – wir vergleichen den gefundenen Datei-Namen mit dem Namen, den wir suchen – und damit wir was finden, suchen wir erstmal nach „test.txt“. Wenn die Datei stimmt, dann geben wir sie mit komplettem Pfad aus. Und bei Verzeichnissen machen wir erstmal nix. import java.io.File; public class Kap_09_07_Bsp_03_FileSucheRekursivStep2 { public static void main(String[] args) { String searchfile = "test.txt"; File path = new File("C:\\JavaDateiSystemBeispiele"); String[] filenames = path.list(); for (String filename : filenames) { File file = new File(path, filename); String fullfilename = file.getAbsolutePath(); if (file.isDirectory()) { // Im Augenblick machen wir hier noch nichts } else if (file.isFile()) { if (file.getName().equals(searchfile)) { System.out.println("-> " + fullfilename); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 184 / 409 } } } } } Ausgabe -> C:\JavaDateiSystemBeispiele\test.txt Kommen wir nun zum Fall, dass das Verzeichnis selber wieder ein Verzeichnis enthält. In diesem Fall müssen wir nur für dieses Verzeichnis das Gleiche machen: alle Elemente durchlaufen, Dateien vergleichen, bei Verzeichnissen wieder das Gleiche. Das klingt nach Rekursion, wie wir sie in Kapitel 8.7 kennen gelernt haben. Wichtig – dazu müssen wir die Funktionalität in eine Funktion auslagern – sonst können wir sie nicht rekursiv aufrufen. Und mehr machen wir in diesem Schritt auch nicht. import java.io.File; public class Kap_09_07_Bsp_04_FileSucheRekursivStep3 { public static void main(String[] args) { String searchpath = "C:\\JavaDateiSystemBeispiele"; String searchfile = "test.txt"; searchFile(searchpath, searchfile); } public static void searchFile(String searchpath, String searchfile) { File path = new File(searchpath); String[] filenames = path.list(); for (String filename : filenames) { File file = new File(path, filename); String fullfilename = file.getAbsolutePath(); if (file.isDirectory()) { // Im Augenblick machen wir hier noch nichts } else if (file.isFile()) { if (file.getName().equals(searchfile)) { System.out.println("-> " + fullfilename); } } } } } Ausgabe -> C:\JavaDateiSystemBeispiele\test.txt Nachdem unsere Verzeichnis-Durchforste-Funktionalität nun in einer Funktion ist, können wir sie jetzt auch aufrufen – und diesmal auch mit dem richtigen Datei-Namen. import java.io.File; public class Kap_09_07_Bsp_05_FileSucheRekursivStep4 { public static void main(String[] args) { String searchpath = "C:\\JavaDateiSystemBeispiele"; String searchfile = "find.txt"; searchFile(searchpath, searchfile); } public static void searchFile(String searchpath, String searchfile) { File path = new File(searchpath); String[] filenames = path.list(); for (String filename : filenames) { File file = new File(path, filename); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 185 / 409 String fullfilename = file.getAbsolutePath(); if (file.isDirectory()) { searchFile(fullfilename, searchfile); } else if (file.isFile()) { if (file.getName().equals(searchfile)) { System.out.println("-> " + fullfilename); } } } } } Mögliche Ausgabe (andere Reihenfolge möglich – ist nicht definiert) -> C:\JavaDateiSystemBeispiele\verzeichnis1\verzeichnis3\verzeichnis4\find.txt -> C:\JavaDateiSystemBeispiele\verzeichnis1\verzeichnis5\find.txt -> C:\JavaDateiSystemBeispiele\verzeichnis2\find.txt Wow – alles funktioniert. So einfach und elegant ist Rekursion. Damit unser Quelltext noch etwas mehr nach richtigem Programm aussieht, integrieren wir noch ein paar Dinge: • Programm-Name • Rückmeldung, wo gesucht wird • Rückmeldung, wonach gesucht wird • Und eine Sicherheits-Abfrage, falls jemand „searchFile“ nicht mit einem Verzeichnis als ersten Parameter aufruft. import java.io.File; public class Kap_09_07_Bsp_06_FileSucheRekursiv { public static void main(String[] args) { System.out.println("Rekursive Datei-Suche mit java.io.File"); String searchpath = "C:\\JavaDateiSystemBeispiele"; String searchfile = "find.txt"; System.out.println("Suche in: " + searchpath); System.out.println("Nach: " + searchfile); searchFile(searchpath, searchfile); } public static void searchFile(String searchpath, String searchfile) { File path = new File(searchpath); if (!path.isDirectory()) { System.out.println("Fehler - kein Verzeichnis"); return; } String[] filenames = path.list(); for (String filename : filenames) { File file = new File(searchpath, filename); String fullfilename = file.getAbsolutePath(); if (file.isDirectory()) { searchFile(fullfilename, searchfile); } else if (file.isFile()) { if (searchfile.equals(filename)) { System.out.println("-> " + fullfilename); } } } } } Mögliche Ausgabe (andere Reihenfolge möglich – ist nicht definiert) Rekursive Datei-Suche mit java.io.File © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 186 / 409 Suche in: c:\aaa Nach: find.txt -> C:\JavaDateiSystemBeispiele\verzeichnis1\verzeichnis3\verzeichnis4\find.txt -> C:\\JavaDateiSystemBeispiele\verzeichnis1\verzeichnis5\find.txt -> C:\JavaDateiSystemBeispiele\verzeichnis2\find.txt Hinweise: • In den Aufgaben 9.8.7 und 9.8.8 werden wir dieses Programm zur rekursiven Datei-Suche um weitere Fähigkeiten ergänzen. • Im nächsten Kapitel wird die rekursive Datei-Suche mit dem JDK 1.7 Package „java.nio.file“ implementiert. 9.7.2 Rekursive Datei-Suche mit „java.nio.file“ Seit dem JDK 1.4 gibt es das Package New-IO „java.nio“ in Java. In Java 7 wurde es um das Unter-Package „java.nio.file“ ergänzt. Hierdrin sind viele Klassen mit stark erweiterten Möglichkeiten gegenüber „java.io.file“ enthalten – über Kopieren von Dateien und Verzeichnissen bis hin zu Themen wie User-Rechte auf Datei-Systeme. So leistungsfähig das neue „java.nio.file“ Package auch ist – die Nutzung ist für Einsteiger oft schwer und komplex. Von daher nutzen wir hier im Java-Tutorial nur die File-Klasse – später sollten Sie sich aber ruhig mal an „java.nio.file“ erinnern, falls Sie aufwändigere Aufgaben im Kontext Datei-System lösen müssen. Als Beispiel, aber ohne jede Erklärung, nochmal die rekursive Datei-Suche – nun aber mit „java.nio.file“. import import import import import import java.io.File; java.io.IOException; java.nio.file.DirectoryStream; java.nio.file.Files; java.nio.file.Path; java.nio.file.Paths; public class Kap_09_07_Bsp_03_FileSucheRekursivMitNiofile { public static void main(String[] args) { System.out.println("Rekursive Datei-Suche mit java.nio.file"); String searchpath = "C:\\JavaDateiSystemBeispiele"; String searchfile = "find.txt"; System.out.println("Suche in: " + searchpath); System.out.println("Nach: " + searchfile); try { Path path = Paths.get(searchpath); searchFile(path, searchfile); } catch (IOException e) { System.out.println("Fehler: " + e.getMessage()); } } public static void searchFile(Path path, String searchfile) throws IOException { DirectoryStream<Path> dirStream = Files.newDirectoryStream(path); for (Path entry : dirStream) { File file = entry.toFile(); if (file.isDirectory()) { searchFile(entry, searchfile); } else if (file.isFile()) { © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 187 / 409 if (file.getName().equals(searchfile)) { System.out.println("-> " + file.getAbsolutePath()); } } } } } Ausgabe (unter Annahme der obigen Datei-Struktur) Rekursive Datei-Suche mit java.nio.file Suche in: c:\aaa Nach: find.txt -> C:\JavaDateiSystemBeispiele\verzeichnis1\verzeichnis3\verzeichnis4\find.txt -> C:\JavaDateiSystemBeispiele\verzeichnis1\verzeichnis5\find.txt -> C:\JavaDateiSystemBeispiele\verzeichnis2\find.txt 9.8 Aufgaben 9.8.1 Aufgabe „String-Analyse“ Schreiben Sie ein Programm das eine Zeile einliest, und ausgibt wie oft welches Zeichen in der Zeile vorkommt. Erzeugen Sie eine Ausgabe der Zeichen in der Reihenfolge ihres ersten Vorkommens im Eingabetext – weitere Vorkommen der Zeichen werden bei der Ausgabe ignoriert. Mögliche Ein- und Ausgabe: Eingabe: Hallo Welt: 123332 'H': 1 'a': 1 'l': 3 'o': 1 ' ': 2 'W': 1 'e': 1 't': 1 ':': 1 '1': 1 '2': 2 '3': 3 Implementieren Sie zwei Lösungen: • Die erste Lösung soll ohne Container auskommen und nur mit den Klassen String, StringBuffer und/oder StringBuilder implementiert werden. • Im Gegensatz zur ersten Lösung sollen Sie hier Container nutzen – und erreichen damit (hoffentlich) eine viel einfachere Lösung. Lösung siehe Kapitel 9.9. 9.8.2 Aufgabe „Hallo <Person>“ Schreiben Sie ein Programm, dass einen kompletten Namen einliest, und diesen Namen mit Sternchen und Leerzeichen umrandet als Anrede ausgibt. Bsp: der eingebene Name ist „Detlef Wilkening“, dann soll folgende Ausgabe erzeugt werden: Mögliche Ein- und Ausgabe: © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 188 / 409 Name: Detlef Wilkening *************************** * * * Hallo Detlef Wilkening! * * * *************************** Lösung siehe Kapitel 9.10. 9.8.3 Aufgabe „Hallo <Personen>“ Schreiben Sie ein Programm ähnlich Aufgabe 9.8.2, dass hier aber beliebig viele komplette Namen einliest. Wird eine leere Eingabe gemacht, so gelten damit alle Namen als eingegeben. Danach sollen alles diese Namen mit Anrede und umrandet mit Sternchen und Leerzeichen ausgegeben werden – die Reihenfolge soll hierbei der Eingabe entsprechen. Die Umrandung soll sich am längsten Namen orientieren. Bsp: die eingegebenen Namen seien „Max“, „Johannes Wolfgang“ und „Werner“, dann soll folgende Ausgabe erzeugt werden: Mögliche Ein- und Ausgabe: Name: Max Name: Johannes Wolfgang Name: Werner Name: **************************** * * * Hallo Max! * * Hallo Johannes Wolfgang! * * Hallo Werner! * * * **************************** Lösung siehe Kapitel 9.11. 9.8.4 Aufgabe „Lesbare Zahlen 2“ Eine ähnliche Aufgabe wie 7.9.3: schreiben Sie ein Programm, dass Benutzer-Eingaben von Ziffern von 1 bis 9 als lesbaren Text ausgibt. Die Eingabe einer beliebigen anderen Zahl oder von etwas anderem sollen das Programm beenden. Bitte geben Sie eine Ziffer ein: 2 -> zwei Bitte geben Sie eine Ziffer ein: 7 -> sieben Bitte geben Sie eine Ziffer ein: x Ende Als Unterschied zu Aufgabe 7.9.3 sollen Sie hier keine Kontrollstruktur (switch oder if) zur Abbildung der Zahlen auf Texte verwenden, sondern einen Container. Machen Sie sich klar, dass diese Lösung kürzer, performanter, und auch noch besser lesbar ist. Die gleiche Aufgabe finden Sie noch mal in Kap. 10.8.1 – dort soll sie mit Arrays gelöst werden, was in diesem Fall noch ein bisschen effizienter ist. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 189 / 409 Lösung siehe Kapitel 9.12. 9.8.5 Aufgabe „Lottozahlen“ Schreiben Sie ein Lotto-Programm, d.h. ein Programm dass sechs Zufallszahlen zwischen 1 und 49 inkl. sortiert ausgibt. Aber Achtung – keine Zufallszahl darf mehrfach vorkommen. Lösung siehe Kapitel 9.13. 9.8.6 Aufgabe „Telefonbuch“ Schreiben Sie ein erstes ganz ganz einfaches Telefonbuch47. Das Telefonbuch soll Namen und Telefonnummern speichern – am Anfang begnügen wir uns mit einer Nummer pro Name, und es soll keine Name doppelt vorkommen (d.h. jeder Name im Telefonbuch ist ein Unikat). Die Telefon-Nummern sind keine Zahlen, sondern Texte, damit z.B. Leerzeichen oder ein Slash „/“ für eine lesbarere Formatierung möglich sind. Die Einträge im Telefonbuch sollen im Augenblick noch fest im Programm stehen, d.h. es können im Augenblick keine Einträge hinzugefügt, gelöscht, oder editiert werden. Der Benutzer kann entweder einen der Befehle „end“ bzw. „list“ oder einen Namen eingeben. • Bei „end“ soll das Programm beendet werden. • Bei „list“ sollen alle Einträge im Telefonbuch (zuerst Name, dann Nummer, eine Zeile pro Eintrag – sortiert nach Namen (nicht lexikalisch sortiert, sondern nach Zeichenkodierung) ausgegeben werden. • Alle anderen Benutzer-Eingaben werden als Namen interpretiert, und im Telefonbuch gesucht. Wird der Name gefunden, so wird die Nummer ausgegeben. Wird der Name nicht gefunden, so wird eine entsprechende Meldung ausgegeben. Achtung – der Name muss exakt übereinstimmen, und es werden auch Groß- und Klein-Schreibung unterschieden. Nach der Bearbeitung des Befehls „list“ oder eines Namens kann der Benutzer die nächste Eingabe machen. Mögliche Ein- und Ausgabe Telefonbuch Eingabe: list 6 - Eintraege: Bernd => 0555 / 55 55 55 Detlef => 0123 / 12 21 1 Dietmar => 0321 / 888 222 99 Edgar => 0111 / 11 11 1 Edmund => 0222 / 33 22 11 Karl => 0111 / 11 22 33 Wir werden dieses Telefonbuch in weiteren Aufgaben mit mehr Wissen zu einer kleinen Kontaktdaten Verwaltung ausbauen, aber hier und jetzt fangen wir erstmal klein an. Für später siehe die Aufgaben 11.7, 11.8, 14.19, 14.16.4, todo oder todo. 47 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 190 / 409 Eingabe: Max Name "Max" ist nicht im Telefonbuch vorhanden. Eingabe: Detlef Detlef => 0123 / 12 21 1 Eingabe: end Programmende Lösung siehe Kapitel 9.14. 9.8.7 Aufgabe „Platten-Platz Verbrauch“ In Kapitel 9.7.1 haben wir eine einfache Form der rekursiven Datei-Suche kennengelernt. Nun sollen Sie das Programm etwas erweitern. • Ihr Programm soll berechnen, wieviel Platten-Platz alle Dateien eines Typs (einer Extension) innerhalb eines Verzeichnisses verbrauchen. • Zusätzlich soll der Benutzer das Verzeichnis und die Such-Extension (ohne „.“) auf der Kommando-Zeile eingeben können. • Die Verzeichnis-Eingabe soll sich so oft wiederholen, bis der Benutzer ein gültiges Verzeichnis eingegeben hat. • Wird keine Extension eingegeben, wird der Platten-Platz Verbrauch aller Dateien im Verzeichnis berechnet. • Vergleichbar zur rekursiven Datei-Suche in Kapitel 9.7.1 ignorieren wir Probleme, die sich aus User-Rechten und fehlenden Zugriffs-Rechten resultieren. Beispiele, unter der Berücksichtigung der Datei-System-Struktur aus Kapitel 9.7.1, und entsprechender Datei-Größen: Mögliche Ein- und Ausgabe Platten-Platz Verbrauch Geben Sie bitte das Verzeichnis ein: C:\JavaDateiSystemBeispiele Geben Sie bitte die Extension ein: dat Gesamtverbrauch: 3472 Bytes Mögliche Ein- und Ausgabe Platten-Platz Verbrauch Geben Sie bitte das Verzeichnis ein: C:\JavaDateiSystemBeispiele Geben Sie bitte die Extension ein: txt Gesamtverbrauch: 5683 Bytes Mögliche Ein- und Ausgabe Platten-Platz Verbrauch Geben Sie bitte das Verzeichnis ein: C:\JavaDateiSystemBeispiele Geben Sie bitte die Extension ein: Gesamtverbrauch: 9155 Bytes © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 191 / 409 Mögliche Ein- und Ausgabe Platten-Platz Verbrauch Geben Sie bitte das Verzeichnis ein: kein-verzeichnis "kein-verzeichnis" ist kein Verzeichnis Geben Sie bitte das Verzeichnis ein: immer noch nicht "immer noch nicht" ist kein Verzeichnis Geben Sie bitte das Verzeichnis ein: C:\JavaDateiSystemBeispiele Geben Sie bitte die Extension ein: dat Gesamtverbrauch: 3472 Bytes Lösung siehe Kapitel 9.15. 9.8.8 Aufgabe „Datei-Suche“ Die nächste Aufgabe erweitert die einfache Datei-Suche aus Kapitel 9.7.1. • Ihr Programm soll alle Dateien innerhalb eines Verzeichnisses suchen, die einem von mehreren Namen entsprechen. • Der Benutzer soll das Such-Verzeichnis und die Such-Datei-Namen auf der KommandoZeile eingeben können. • Die Such-Verzeichnis-Eingabe soll sich so oft wiederholen, bis der Benutzer ein gültiges Verzeichnis eingegeben hat. • Die Such-Datei-Namen-Eingabe soll sich so oft wiederholen, bis der Benutzer einen Leerstring eingibt. Alle Namen bis dahin gelten als zu Suchen. • Die gefundenen Dateien mit ihren Pfaden sollen sortiert nach den Datei-Namen ausgeben werden. Jeder Ausgabe-Block soll mit dem Datei-Namen beginnen, und danach eine eingerückte Aufzählung der Verzeichnisse (inkl. Datei-Namen). • Wurde keine Datei zu einem Datei-Such-Namen gefunden, so wird nur der Datei-SuchName ausgegeben. • Vergleichbar zur rekursiven Datei-Suche in Kapitel 9.7.1 ignorieren wir Probleme, die sich aus User-Rechten und fehlenden Zugriffs-Rechten resultieren. Beispiele, unter der Berücksichtigung der Datei-System-Struktur aus Kapitel 9.7.1: Mögliche Ein- und Ausgabe Datei-Suche ----------Geben Geben Geben Geben Sie Sie Sie Sie bitte bitte bitte bitte das die die die Such-Verzeichnis ein: C:\JavaDateiSystemBeispiele Such-Namen ein: find.txt Such-Namen ein: search.dat Such-Namen ein: find.txt -> C:\JavaDateiSystemBeispiele\verzeichnis1\verzeichnis3\verzeichnis4\find.txt -> C:\JavaDateiSystemBeispiele\verzeichnis1\verzeichnis5\find.txt -> C:\JavaDateiSystemBeispiele\verzeichnis2\find.txt search.dat -> C:\JavaDateiSystemBeispiele\search.dat -> C:\JavaDateiSystemBeispiele\verzeichnis1\verzeichnis3\verzeichnis4\search.dat Mögliche Ein- und Ausgabe Datei-Suche © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 192 / 409 ----------Geben Sie bitte das Such-Verzeichnis ein: kein Verzeichnis "kein Verzeichnis" ist kein Verzeichnis Geben Geben Geben Geben Geben Sie Sie Sie Sie Sie bitte bitte bitte bitte bitte das die die die die Such-Verzeichnis ein: C:\JavaDateiSystemBeispiele Such-Namen ein: search.dat Such-Namen ein: search.txt Such-Namen ein: search.png Such-Namen ein: search.dat -> C:\JavaDateiSystemBeispiele\search.dat -> C:\JavaDateiSystemBeispiele\verzeichnis1\verzeichnis3\verzeichnis4\search.dat search.png search.txt Lösung siehe Kapitel 9.16. 9.9 Lsg. zu Aufgabe „String-Analyse“ – Kap. 9.8.1 9.9.1 Erste Lösung mit String, StringBuffer bzw. StringBuilder Als erstes müssen wir einen String von der Kommandozeile einlesen. Da dies – dank Reader und Exceptions – kein Einzeiler ist, lagern wir dies in eine Funktion aus. Im Fehlerfall gibt die Funktion einen Leerstring zurück – damit können wir im Augenblick gut leben. So könnte die Funktion aussehen48. public static String readLine() { String in = ""; try { InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); in = reader.readLine(); } catch (Exception x) { } return in; } Dann kommt die eigentlich Analyse des Strings. Dazu müssen wir über den String laufen, und jeden Buchstaben auf die Anzahl der Vorkommen überprüfen – der Pseudocode sieht im ersten Schritt also so aus. Achtung, dieser Algorithmus ist noch fehlerhaft. // Vorsicht - noch fehlerhafter Algorithmus for-each zeichen in string laufe ueber string und zaehle anzahl dieses zeichens zeichen und anzahl ausgeben In Java-Code könnte das so aussehen: // Vorsicht - noch fehlerhafter Algorithmus for (int i=0; i<in.length(); i++) { char c = in.charAt(i); int count = 0; Die Funktion ist übrigens so gut, daß wir sie in den nächsten Aufgaben auch immer wieder ohne lange Erklärung wiederbenutzen werden. 48 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 193 / 409 for (int j=0; j<in.length(); j++) { if (in.charAt(j)==c) { count++; } } System.out.println("'" + c + "': " + count); } Ausgabe bei in = "aba" 'a': 2 'b': 1 'a': 2 Das Problem ist, dass wir jetzt die mehrfach vorkommenden Zeichen auch mehrfach zählen und ausgeben – und das ist nicht gewollt. Wir müssen also die Zeichen markieren, die wir schon gezählt haben. Da wir diese Lösung ohne Container implementieren wollen, müssen wir im String vermerken, wann ein Zeichen schon behandelt worden ist. Dazu brauchen wir ein Zeichen, dass im String nicht auftauchen kann – also als eindeutiges „Schon-Behandelt-Zeichen“ dienen kann. Im Rahmen unserer Aufgabe gibt es ein solches Zeichen: der Zeilenumbruch ‚\n‘. Dieses Zeichen kann nicht im String vorhanden sein, da wir den String von der Tastatur einlesen, und die Return-Taste die Eingabe abschliesst. Damit wir auf dieses Marker-Zeichen gut lesbar im Code einsetzen können, benutzen wir es nicht direkt im Quelltext, sondern nutzen eine lokale Konstante, d.h. eine lokale Variable mit dem Modifier „final“ – siehe Kapitel 8.3 und auch Kapitel 7.13.1. Damit wird aus unserem Code von oben: final char ok = '\n'; for (int i=0; i<in.length(); i++) { char c = in.charAt(i); if (c==ok) continue; // (*) int count = 0; for (int j=0; j<in.length(); j++) { if (in.charAt(j)==c) { count++; in.setCharAt(j, ok); // (**) } } System.out.println("'" + c + "': " + count); } Neben der lokalen Konstanten hat sich der Code an zwei Stellen geändert: • Ist das zu analysierende Zeichen unser Marker-Zeichen, so dürfen wir nicht analysieren, und setzen die Schleife direkt fort – Zeile (*). • Finden wir während der Analyse im String das zu suchende Zeichen, dann müssen wir es durch das Marker-Zeichen ersetzen – Zeile (**). Damit funktioniert unser Programm, und wir könnten uns zufrieden der nächsten Aufgabe widmen. Aber so ganz optimal ist unser Programm noch nicht. Sobald wir ein neu zu zählendes Zeichen finden, wissen wir ja, dass es im String vorher nicht vorhanden sein kann – ansonsten hätten wir es ja schon gezählt. Also brauchen wir den Anfang des Strings für das Zählen nicht © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 194 / 409 zu berücksichtigen. Wenn wir mit dem Zählwert „count=1“ starten, brauchen wir erst mit dem Zeichen „j+1“ für das Zählen zu starten. Da eine For-Schleife kopfgesteuert ist, und daher auch 0-mal durchlaufen werden kann (siehe Kapitel 7.3), funktioniert dies auch problemlos für das letzte Zeichen der Schleife. final char ok = '\n'; for (int i=0; i<in.length(); i++) { char c = in.charAt(i); if (c==ok) continue; int count = 1; for (int j=i+1; j<in.length(); j++) { if (in.charAt(j)==c) { count++; in.setCharAt(j, ok); } } System.out.println("'" + c + "': " + count); } Aber seien Sie sich darüber im klaren – auch mit unserer Optimierung ist dies kein schneller Algorithmus. Im Prinzip muss für jedes Zeichen im String der String einmal vollständig durchlaufen werden. Man nennt dies einen quadratischen Algorithmus, da er eine quadratische Zeit-Komplexität hat. Eine Verdopplung der String-Länge führt zu einer Vervierfachung der notwendigen Zeit, eine Verdreifachung der String-Länge zu einer Verneunfachung der notwendigen Zeit. Besser wird die Lösung mit Container im nächsten Kapitel sein. Mit Klasse und Main-Funktion ergibt das alles zusammen folgenden Code: import java.io.BufferedReader; import java.io.InputStreamReader; public class Appl { public static String readLine() { String in = ""; try { InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); in = reader.readLine(); } catch (Exception x) { } return in; } public static void main(String[] args) { final char ok = '\n'; System.out.print("Eingabe: "); StringBuilder in = new StringBuilder(readLine()); for (int i=0; i<in.length(); i++) { char c = in.charAt(i); if (c==ok) continue; int count = 1; for (int j=i+1; j<in.length(); j++) { if (in.charAt(j)==c) { count++; in.setCharAt(j, ok); } } System.out.println("'" + c + "': " + count); } } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 195 / 409 } Ausgabe Eingabe: Hallo Waldilal 'H': 1 'a': 3 'l': 5 'o': 1 ' ': 1 'W': 1 'd': 1 'i': 1 9.9.2 Zweite Lösung mit Container Erklärung folgt (hoffentlich) später – todo... import import import import import java.io.BufferedReader; java.io.InputStreamReader; java.util.ArrayList; java.util.Iterator; java.util.TreeMap; public class Appl { public static String readLine() { String in = ""; try { InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); in = reader.readLine(); } catch (Exception x) { } return in; } public static void main(String[] args) { System.out.print("Eingabe: "); String in = readLine(); ArrayList<Character> order = new ArrayList<>(); TreeMap<Character, Integer> map = new TreeMap<>(); for (int i = 0; i < in.length(); i++) { char c = in.charAt(i); Integer count = map.get(c); if (count == null) { map.put(c, 1); order.add(c); } else { map.put(c, ++count); } } for (char c : order) { int count = map.get(c); System.out.println("'" + c + "': " + count); } } } Ausgabe Eingabe: Hallo Waldilal 'H': 1 'a': 3 'l': 5 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 'o': ' ': 'W': 'd': 'i': Seite 196 / 409 1 1 1 1 1 9.10 Lsg. zu Aufgabe „Hallo <Person>“ – Kap. 9.8.2 Erklärung folgt (hoffentlich) später – todo... import java.io.BufferedReader; import java.io.InputStreamReader; public class Appl { public static String readLine() { String in = ""; try { InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); in = reader.readLine(); } catch (Exception x) { } return in; } public static String string(int n, char c) { StringBuilder erg = new StringBuilder(n); for (int i=0; i<n; i++) { erg.append(c); } return erg.toString(); } public static void main(String[] args) { System.out.print("Bitte geben Sie Ihren Namen ein: "); String name = readLine(); int len = name.length() + 11; System.out.println(string(len, '*')); System.out.println('*' + string(len-2, ' ') + '*'); System.out.println("* Hallo " + name + "! *"); System.out.println('*' + string(len-2, ' ') + '*'); System.out.println(string(len, '*')); } } Ausgabe Bitte geben Sie Ihren Namen ein: Detlef Wilkening *************************** * * * Hallo Detlef Wilkening! * * * *************************** 9.11 Lsg. zu Aufgabe „Hallo <Personen>“ – Kap. 9.8.3 Erklärung folgt (hoffentlich) später – todo... import java.io.BufferedReader; import java.io.InputStreamReader; import java.util.ArrayList; © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 197 / 409 import java.util.Iterator; public class Appl { public static String readLine() { String in = ""; try { InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); in = reader.readLine(); } catch (Exception x) { } return in; } public static String string(int n, char c) { StringBuilder erg = new StringBuilder(n); for (int i=0; i<n; i++) { erg.append(c); } return erg.toString(); } public static void main(String[] args) { ArrayList<String> list = new ArrayList<>(); int maxLen = 0; while (true) { System.out.print("Bitte geben Sie die Namen ein: "); String name = readLine(); if (name.length()==0) break; list.add(name); if (name.length()>maxLen) { maxLen = name.length(); } } int len = maxLen + 11; String stars = string(len, '*'); String spaces = string(len - 2, ' '); System.out.println(stars); System.out.println('*' + spaces + '*'); for (String s : list) { int slen = s.length(); String rest = string(maxLen - slen + 1, ' '); System.out.println("* Hallo " + s + '!' + rest + '*'); } System.out.println('*' + spaces + '*'); System.out.println(stars); } } Ausgabe Bitte geben Sie die Namen Bitte geben Sie die Namen Bitte geben Sie die Namen Bitte geben Sie die Namen *********************** * * * Hallo Max! * * Hallo Wolf-Guenter! * * Hallo Tassilo! * * * *********************** ein: Max ein: Wolf-Guenter ein: Tassilo ein: © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 198 / 409 9.12 Lsg. zu Aufgabe „Lesbare Zahlen 2“ – Kap. 9.8.4 9.12.1 Lösung 1: Doch mit Switch-Anweisung Erklärung folgt (hoffentlich) später – todo... import java.io.BufferedReader; import java.io.InputStreamReader; public class Appl { public static String readLine() { String in = ""; try { InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); in = reader.readLine(); } catch (Exception x) { } return in; } public static void main(String[] args) { while (true) { System.out.print("Bitte geben Sie eine Ziffer ein: "); String in = readLine(); int number; try { number = Integer.parseInt(in); if (number<1 || number>9) break; } catch (Exception x) { break; } switch (number) { case 1: System.out.println("-> break; case 2: System.out.println("-> break; case 3: System.out.println("-> break; case 4: System.out.println("-> break; case 5: System.out.println("-> break; case 6: System.out.println("-> break; case 7: System.out.println("-> break; case 8: System.out.println("-> break; case 9: System.out.println("-> break; } eins"); zwei"); drei"); vier"); fuenf"); sechs"); sieben"); acht"); neun"); } System.out.println("Ende"); } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 199 / 409 } 9.12.2 Lösung 2: Mit TreeMap Erklärung folgt (hoffentlich) später – todo... import java.io.BufferedReader; import java.io.InputStreamReader; import java.util.HashMap; public class Appl { public static String readLine() { String in = ""; try { InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); in = reader.readLine(); } catch (Exception x) { } return in; } public static void main(String[] args) { // HashMap ist sehr gut, aber nicht optimal – noch besser ist Loesung 3 HashMap<Integer, String> numbers = new HashMap<>(); numbers.put(0, "null"); numbers.put(1, "eins"); numbers.put(2, "zwei"); numbers.put(3, "drei"); numbers.put(4, "vier"); numbers.put(5, "fuenf"); numbers.put(6, "sechs"); numbers.put(7, "sieben"); numbers.put(8, "acht"); numbers.put(9, "neun"); while (true) { System.out.print("Bitte geben Sie eine Ziffer ein: "); String in = readLine(); int number; try { number = Integer.parseInt(in); if (number < 1 || number > 9) { break; } } catch (Exception x) { break; } System.out.println(numbers.get(number)); } } } 9.12.3 Lösung 3: Mit ArrayList Erklärung folgt (hoffentlich) später – todo... import java.io.BufferedReader; import java.io.InputStreamReader; import java.util.ArrayList; © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 200 / 409 public class Appl { public static String readLine() { String in = ""; try { InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); in = reader.readLine(); } catch (Exception x) { } return in; } public static void main(String[] args) { final ArrayList<String> numbers = new ArrayList<>(); numbers.add("null"); numbers.add("eins"); numbers.add("zwei"); numbers.add("drei"); numbers.add("vier"); numbers.add("fuenf"); numbers.add("sechs"); numbers.add("sieben"); numbers.add("acht"); numbers.add("neun"); while (true) { System.out.print("Bitte geben Sie eine Ziffer ein: "); String in = readLine(); int number; try { number = Integer.parseInt(in); if (number < 0 || number > 9) { break; } } catch (Exception x) { break; } System.out.println(numbers.get(number)); } } } 9.13 Lsg. zu Aufgabe „Lottozahlen“ – Kap. 9.8.5 Erklärung folgt (hoffentlich) später – todo... import java.util.Iterator; import java.util.Random; import java.util.TreeSet; public class Appl { public static void main(String[] args) { Random rnd = new Random(); TreeSet<Integer> numbers = new TreeSet<>(); while (numbers.size() < 6) { numbers.add(rnd.nextInt(49) + 1); } for (Integer i : numbers) { System.out.print(i + " "); } } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 201 / 409 } Mögliche Ausgabe 7 13 28 33 34 44 Sehr sehr wichtiger Hinweis – wenn Sie mit den Zahlen dieses Programms Lotto spielen und gewinnen, dann müssen Sie mich mit 40% am Gewinn beteiligen. Benachrichtigen Sie mich bitte per Mail „detlef@wilkening-online.de“. 9.14 Lsg. zu Aufgabe „Telefonbuch“ – Kap. 9.8.6 Erklärung folgt (hoffentlich) später – todo... import import import import java.io.BufferedReader; java.io.InputStreamReader; java.util.Map.Entry; java.util.TreeMap; public class Appl { public static String readLine() { String in = ""; try { InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); in = reader.readLine(); } catch (Exception x) { } return in; } public static void main(String[] args) { System.out.println("Telefonbuch"); TreeMap<String, String> items = new TreeMap<>(); items.put("Detlef", "0123 / 12 21 1"); items.put("Edgar", "0111 / 11 11 1"); items.put("Bernd", "0555 / 55 55 55"); items.put("Edmund", "0222 / 33 22 11"); items.put("Karl", "0111 / 11 22 33"); items.put("Dietmar", "0321 / 888 222 99"); while (true) { System.out.print("\nEingabe: "); String in = readLine(); System.out.println(); if (in.equals("end")) break; if (in.equals("list")) { System.out.println(items.size() + " Eintraege:"); for (Entry<String, String> e : items.entrySet()) { System.out.println("- " + e.getKey() + " : " + e.getValue()); } continue; } String number = items.get(in); if (number == null) { System.out.println( "Name \"" + in + "\" ist nicht im Telefonbuch vorhanden."); continue; } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 202 / 409 System.out.println(in + " => " + number); } System.out.println("Programmende"); } } 9.15 Lsg. zu Aufgabe „Platten-Platz Verbrauch“ – Kap. 9.8.7 Erklärung folgt (hoffentlich) später – todo... import java.io.BufferedReader; import java.io.File; import java.io.InputStreamReader; public class Kap_09_15_Lsg_PlattenPlatzVerbrauch { public static void main(String[] args) { System.out.println("Platten-Platz Verbrauch\n"); File dir = readDirectory(); String ext = readSearchExtension(); long size = calcFileSizes(dir, ext); System.out.println("\nGesamtverbrauch: " + size + " Bytes"); } public static String readLine() { String in = ""; try { InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); in = reader.readLine(); } catch (Exception x) { } return in; } public static File readDirectory() { while (true) { System.out.print("Geben Sie bitte das Verzeichnis ein: "); String in = readLine(); File dir = new File(in); if (dir.isDirectory()) { return dir; } System.out.println("\"" + dir + "\" ist kein Verzeichnis\n"); } } public static String readSearchExtension() { System.out.print("Geben Sie bitte die Extension ein: "); String in = readLine(); if (in.isEmpty()) { return in; } return "." + in; } public static long calcFileSizes(File dir, String extension) { long res = 0; String[] filenames = dir.list(); for (String filename : filenames) { File file = new File(dir, filename); if (file.isDirectory()) { res += calcFileSizes(file, extension); } else if (file.isFile()) { © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 203 / 409 if (file.getName().endsWith(extension)) { res += file.length(); } } } return res; } } 9.16 Lsg. zu Aufgabe „Datei-Suche“ – Kap. 9.8.8 Erklärung folgt (hoffentlich) später – todo... import import import import import import java.io.BufferedReader; java.io.File; java.io.InputStreamReader; java.util.ArrayList; java.util.Map.Entry; java.util.TreeMap; public class Kap_09_16_Lsg_DateiSuche { public static void main(String[] args) { System.out.println("Datei-Suche\n-----------\n"); File dir = readDirectory(); TreeMap<String, ArrayList<String>> files = readSearchFileNames(); searchFile(dir, files); System.out.println(); for (Entry<String, ArrayList<String>> entry : files.entrySet()) { System.out.println(entry.getKey()); for (String ffn : entry.getValue()) { System.out.println("-> " + ffn); } } } public static String readLine() { String in = ""; try { InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); in = reader.readLine(); } catch (Exception x) { } return in; } public static File readDirectory() { while (true) { System.out.print("Geben Sie bitte das Such-Verzeichnis ein: "); String in = readLine(); File dir = new File(in); if (dir.isDirectory()) { return dir; } System.out.println("\"" + dir + "\" ist kein Verzeichnis\n"); } } public static TreeMap<String, ArrayList<String>> readSearchFileNames() { TreeMap<String, ArrayList<String>> res = new TreeMap<>(); while (true) { System.out.print("Geben Sie bitte die Such-Namen ein: "); String in = readLine(); if (in.isEmpty()) { © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 204 / 409 return res; } res.put(in, new ArrayList<String>()); } } public static void searchFile(File dir, TreeMap<String, ArrayList<String>> files) { String[] filenames = dir.list(); for (String filename : filenames) { File file = new File(dir, filename); if (file.isDirectory()) { searchFile(file, files); } else if (file.isFile()) { findFile(file, files); } } } public static void findFile(File file, TreeMap<String, ArrayList<String>> files) { for (Entry<String, ArrayList<String>> entry : files.entrySet()) { if (file.getName().equals(entry.getKey())) { entry.getValue().add(file.getAbsolutePath()); } } } } 10 Arrays Arrays sind schon kurz in Kapitel 3.5 eingeführt worden. Hier wollen wir sie jetzt detailierter besprechen. • • • • • • • Arrays sind in Java Objekte, d.h. sie stellen keinen „elementaren“ Daten-Typ dar. Array-Variablen sind Referenz-Variablen (siehe 5.4.2). Auf die Array-Elemente kann sowohl lesend als auch schreibend mit dem Operator [int] zugegriffen werden. Array-Zugriffe mit dem Operator [int] sind 0-basiert. Zugriffe auf nicht existente Elemente lösen die Exception „IndexOutOfBoundsException“ aus – siehe Kapitel 10.6. Arrays haben ein Attribut „length”, dass die Anzahl an Elementen im Array enthält. Arrays sind von der Klasse „Object“ abgeleitet – siehe Kapitel 14.11. 10.1 Deklaration Arrays können in Java auf zwei Arten deklariert werden: Die eckigen Klammern, die ein Array als solches identifizieren, können sowohl am Typ als als auch am Bezeichner stehen. Die typische Syntax in Java ist die erste Variante. int[] a1; int a2[]; // Typische Java Syntax // Auch moeglich, sieht man aber selten Initialisierung © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 205 / 409 Es gibt zwei Arten, Arrays zu initialisieren: 1. Literale Initailisierung Bei der literalen Initialisierung wird hinter der Deklaration in geschweiften Klammern eine Liste der Initialwerte aufgeführt. Der Compiler erzeugt implizit ein Array entsprechender Größe und weist die Werte dem Array zu. Es darf dabei keine Größe in den eckigen Klammern der Variablen-Deklaration angegeben werden, auch nicht die Richtige. int[] a = { 1, 2, 3, 4 }; boolean[] b = { true, false, true }; double[2] d = { 1.4, 3.1 }; // Compiler-Error – Groessenangabe nicht erlaubt Das Array a bekommt hier automatisch die Größe 4, das Array b die Größe 3 zugewiesen. Ausserdem werden die Arrays mit den aufgeführten Werten initialisiert. Die literale Initialisierung darf nur bei der Deklaration einer Array-Variablen benutzt werden. Für spätere Neubesetzung muss die dynamische Initialisierung benutzt werden. int[] a = { 1, 2, 3, 4 }; ... a = { 1, 2, 6, 8 }; // Compiler-Error // Hier ist nur die dynamische Initialisierung erlaubt 2. Dynamische Initialisierung Bei der dynamischen Initialisierung legt man selber das Array mit new, Typname und gewünschter Größe in eckigen Klammern an – das Array wird dabei mit den Defaultwerten des Typ‘s initialisiert. int[] a = new int[6]; boolean[] b = new boolean[2]; Hierbei wird mit new ein entsprechender Speicherplatz reserviert (a - Größe 6 / b - Größe 2) und die Array-Elemente werden mit den Default-Werten der jeweiligen Datentypen gefüllt. Auch bei der dynamischen Initialisierung können die Initialisierungswerte in Form einer Liste angegeben werden. Dabei darf keine Größenangabe in den eckigen Klammern stehen und die Liste muss direkt hinter den eckigen Klammern folgen. Die Größe des Arrays bestimmt der Compiler implizit aus der Anzahl an Werten in der Liste. int[] a = new int[] { 2, 5, 9 }; boolean[] b = new boolean[2] { true, false }; // Compiler-Error - wegen Groessenangabe 10.2 Arrays haben ein Attribut: length Array-Objekte haben keine Funktion und genau ein Attribut, das nur lesenden Zugriff erlaubt: ‚length‘ liefert die Anzahl an Elementen im Array zurück. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 206 / 409 int[] a = { 1, 2, 3, 4 }; System.out.println(a.length); Ausgabe 4 10.3 Arrays durchlaufen Seit Java 5 (JDK 1.5) gibt es einen neuen Schleifen-Typ für Container und Arrays, der schon in Kapitel 7.4 kurz vorgestellt wurde. Wollen Sie einfach nur über das gesamte Array laufen, dann nutzen Sie diesen Typ. Alternativ können Sie natürlich ein normale For-Schleife mit SchleifenZähler nutzen. int[] a = { 1, 2, 3, 4 }; System.out.println("Neue Schleife: "); for (int n : a) { System.out.print(n + " "); } System.out.println(); System.out.println("Alte Schleife: "); for (int i=0; i<a.length; i++) { System.out.print(a[i] + " "); } System.out.println(); Ausgabe Neue Schleife: 1 2 3 4 Alte Schleife: 1 2 3 4 10.4 Arrays sind Referenzen Auch Arrays sind keine elementaren Datentypen, d.h. unterliegen auch Array-Variablen der Referenz-Semantik. int[] a = { 1, 2, 3, 4 }; int[] b = a; a[2] = 42; for (int i : a) { System.out.print(i + " } System.out.println(); for (int i : b) { System.out.print(i + " } System.out.println(); Ausgabe 1 2 42 1 2 42 "); "); 4 4 Im Beispiel zeigen beide Array-Variablen auf das gleiche Array. Der schreibende Zugriff auf das dritte Element von a (a[2] = 42;) ändert damit auch das Array b. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 207 / 409 1. Array int[] a: => b: => 1 2. 2 3 4 Array int[] a: => b: => 1 2 42 4 Abb. 10-1 : Referenz-Semantik bei Arrays 10.5 Arrays können mehrdimensional sein Natürlich können Arrays auch mehrdimensional sein. Sie sind kein spezieller Datentyp, sondern werden als Array-von-Arrays implementiert. Die Deklaration der Variablen enthält dann für jede Dimension ein Klammen-Paar, die Initialisierungsliste besteht dann aus einer Liste von Listen. Der Zugriff erfolgt über eine entsprechende Anzahl von Index-Operatoren []. int[][] a = { {1,2}, {3, 4}, {5,6} }; for (int[] aa : a) { for (int i : aa) { System.out.print(i + " "); } System.out.println(); } Ausgabe 1 2 3 4 5 6 Achtung – seien Sie sich darüber im klaren, dass Java die Elemente eines mehrdimensionales Arrays nicht bündig in einem Block im Speicher ablegt, sondern es ein echtes Array-aus-Arrays ist. Array: int[][] 4 Array: int[] 3 > 2 Array: int[] Array: int[] 1 > => > a: 5 6 Abb. 10-2 : Speicher-Layout eines mehrdimensionalen Arrays Arrays müssen nicht unbedingt rechteckig sein © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 208 / 409 Aus dem oben gesagten folgt natürlich auch, dass Arrays nicht unbedingt rechteckig sein müssen. int[][] a = { {1}, {2, 3}, {4, 5, 6, 7} }; for (int[] aa : a) { for (int i : aa) { System.out.print(i + " "); } System.out.println(); } Ausgabe 1 2 3 4 5 6 7 Array: int[][] 3 Array: int[] 2 > Array: int[] Array: int[] 1 > => > a: 4 5 6 7 Abb. 10-3 : Speicher-Layout eines nicht-rechteckigen Arrays 10.6 Fehlerhafter Index Der Zugriff auf ein nicht existentes Objekt führt java-typisch zu einer Exception („java.lang.IndexOutOfBoundsException“). Zugriffe mit fehlerhaftem Index führen also nicht zu potentiellen Problemen wie in manchen anderen Programmier-Sprachen49, sondern werden sauber gemeldet. int[] array = { 1, 2 }; array[0] = 0; // Okay array[1] = 0; // Okay array[2] = 0; // Laufzeit-Fehler => wirft Exception array[-1] = 0; // Laufzeit-Fehler => wirft Exception 10.7 Vergleiche Werden zwei Arrays mit dem Operator „==“ verglichen, so wird nur die Identität verglichen, da es sich um Referenz-Objekte handelt – siehe Kapitel 5.4.2. Auch die Benutzung der von „Object“ geerbten Element-Funktion „equals“ ändert daran nichts, da das Default-Verhalten der Vergleich auf Identität ist – siehe Kapitel todo – und die Elementfunktion für Arrays nicht überschrieben worden ist. 49 Wie z.B. C oder C++. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 209 / 409 Für Werte-Vergleiche auf Arrays muss die Funktion „java.util.Arrays.equals“ benutzt werden. 10.8 Aufgaben 10.8.1 Aufgabe „Lesbare Zahlen 3“ Die gleiche Aufgabe wie 9.8.4: schreiben Sie ein Programm, dass Benutzer-Eingaben von Zahlen von 1 bis 9 als lesbaren Text ausgibt. Die Eingabe einer beliebigen anderen Zahl oder eine Fehleingabe soll das Programm beenden. Bitte geben Sie eine Zahl ein: 2 -> zwei Bitte geben Sie eine Zahl ein: 7 -> sieben Bitte geben Sie eine Zahl ein: x -> Ende Als Unterschied zu Aufgabe 9.8.4 sollen Sie hier keinen Container, sondern ein Array verwenden. Machen Sie sich klar, dass diese Lösung vielleicht noch effizienter ist. Lösung siehe Kapitel 10.9. 10.9 Lsg. zu Aufgabe „Lesbare Zahlen 3“ – Kap. 10.8.1 Erklärung folgt (hoffentlich) später – todo... import java.io.BufferedReader; import java.io.InputStreamReader; public class Appl { public static String readLine() { String in = ""; try { InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); in = reader.readLine(); } catch (Exception x) { } return in; } public static void main(String[] args) { final String[] numbers = { "eins", "zwei", "drei", "vier", "fuenf", "sechs", "sieben", "acht", "neun" }; while (true) { System.out.print("Bitte geben Sie eine Zahl ein: "); String in = readLine(); int number; try { number = Integer.parseInt(in); if (number<1 || number>9) break; } catch (Exception x) { © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 210 / 409 break; } System.out.println("-> " + numbers[number-1]); } System.out.print("-> Ende"); } } 11 Klassen 11.1 Motivation Um nicht-triviale Probleme in den Griff zu bekommen, muss man die Probleme in kleine überschaubare Einheiten zerlegen (Stichwort „teile und herrsche“), die möglichst unabhängig von einander sein sollten (Stichwort „Entkopplung“). Das heißt, man muss in seinem Programm Abstraktionen einziehen, die Komplexität verbergen (am besten so kapseln, dass gar keine Zugriff auf die Interna möglich ist – Stichwort „Information-Hiding“) und auf eine einfache Art Funktionalität zur Verfügung stellen. Ein Beispiel dafür sind z.B. die Container-Klassen von Java, die die gesamten für die Speicherung und Verwaltung von Objekten notwendigen Daten, Techniken und Mechanismen verbergen. In unserem normalen Leben arbeiten wir doch auch so. Wenn sie z.B. ein Haus bauen, dann werden weder sie, noch der Architekt, und wohl auch nicht der Maurer hierbei die Eigenschaften von Elementarteilchen wie Quarks oder Bosonen berücksichtigen. Jeder arbeitet mit den Abstraktions-Ebenen, die für sein Problem angepaßt sind. Der eine denkt halt in Elementarteilchen, dem nächsten passen Atom-Kerne und Elekronenhüllen. Wieder andere denken in Molekülen, noch andere z.B. in Werkstoffeigenschaften. Ein Ingenieur wechselt dann sich in die Sichtweise von – tief unten – Schrauben und Zahnrädern. Weiter oben wird dann z.B. in Getriebe oder Motoren gedacht. Usw, usw. . Jeder arbeitet halt mit den AbstraktionsEbenen, die für sein Problem angepaßt sind. Und alles darunter interessiert ihn nicht, bzw. wird aktiv versucht auszublenden, damit das Problem überschaubar bleibt. Abb. 11-1 : Das Schlüsselwort heißt „Abstraktion“ Einer der Hauptgedanken der Objekt-Orientierung ist daher die Idee, abgeschlossene gekapselte Einheiten programmieren und anbieten zu können, bei denen der Benutzer sich keine Gedanken mehr über das Innenleben machen muss, sondern sie einfach über ihre Schnittstelle benutzt. Abb. 11-2 : Objekt-Kapselung und Benutzung über die Schnittstelle Damit ist das Ziel klar – es muß möglich sein: • abgeschlossene gekapselte Einheiten zu definieren, © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 211 / 409 • diesen Einheiten eine Schnittstelle zu geben, • sie sollte einfach zu implementieren sein, und • einfach zu benutzen sein. Dies wird in Java – wie in allen OO Sprachen – über Klassen und Objekte erreicht. Hierbei sind • Klassen eine allgemeine Beschreibung einer bestimmten Art Objekte. Eine Klasse Auto beschreibt ganz allgemein alle Autos, sowohl meine alte Karre, einen neues Familienauto, als auch das Cabrio meines Kollegen. Allen diesen Autos ist gemein, dass sie Autos sind, und sich durch gemeinsame Fähigkeiten und Merkmale wie Grösse, Leistung, Verbrauch, Farbe, usw. auszeichnen. • Objekte dagegen sind ein konkretes Exemplar (Instanz) einer Klasse. Daher meine alte Karre ist ein konkretes Exemplar der Klasse Auto. Abb. 11-3 : Klassen und Objekte am Beispiel von Autos 11.2 Klassen-Entwurf Bevor sie eine eigene Klasse entwerfen, sollten sie sich immer einige Gedanken über den Sinn, die Repräsentation und die Implementation der Klasse machen. 1. Überlegung - Art der Objekte: • Was für Objekte soll die Klasse repräsentieren? 2. Überlegung - Schnittstelle: • Was soll mit den Objekten der Klasse machbar sein? • Wie sollen sich die Objekte verhalten? • Welche Schnittstelle benötigt die Klasse dafür? 3. Überlegung - Vererbung: • Gibt es Klassen, die eine vergleichbare Schnittstelle haben? • Gibt es Klassen, die ähnliche Funktionalität aufweist? • Gibt es Klassen, die diese Objekte in allgemeinerer Form repräsentieren? • Kann eine abstrakte Schnittstelle konkrete Klassen verbergen? Diese Frage können wir uns erst stellen, wenn wir Vererbung kennen – siehe Kapitel todo. Ich habe sie hier aber aufgeführt, damit sie hier alle relevanten Fragen auf einen Schlag sehen. 4. Überlegung - Implementierung: • Was soll mit den Objekten der Klasse machbar sein? • Wie sollen sich die Objekte verhalten? • Wie implementiere ich diese Funktionalitäten? • Welchen inneren Aufbau wähle ich dafür? Bemerkung – die zweite und vierte Überlegung beruhen auf den gleichen Fragen, aber einmal aus der Sicht eines Benutzers der Klassen, und das andere Mal aus der Sicht des © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 212 / 409 Implementierers gestellt. Bemerkung – in einem größeren Kontext betrachtet würde man die Überlegungen 1 bis 3 als Analyse und die Überlegungen 2 bis 4 als Design betrachten. 11.3 Beispiel 11.3.1 Wurm-Programm Vielleicht kennen sie kleine Gimmick-Programme, die z.B. gerne als Bildschirmschoner eingesetzt werden. Hier konkret geht es um ein Programm, dass einen Wurm über den Bildschirm krabbeln läßt. Dabei soll der Wurm, der den Bildschirm rechts verläßt links wieder hinein krabbeln, und umgekehrt. Verläßt der Wurm den Bildschirm oben, so soll er unten wieder hinein krabbeln, und umgekehrt. Für den Wurm bildet der Bildschirm also eine unendliche Fläche, bei der die jeweils gegenüberliegenden Ränder aufeinander abgebildet werden. Da wir zur Zeit noch kein GUI programmieren können, und auch sonst noch ziemlich viel JavaWerkzeug fehlt, beschränken wir uns hier mit einem ganz kleinen und sehr stark abgespeckten Wurm Programm: • Statt mit einem GUI Fenster arbeiten wir nur mit der Konsole. • Statt eines Wurms wird hier nur ein Zeichen gezeichnet – der Stern ‚*‘. • Statt der Bildschirmbreite setzen wir eine feste Breite von 40 Zeichen an. • Statt mit einer x/y-Koordinate wird hier nur x variiert. • Statt den alten Wurm zu löschen, geben wir eine neue Zeile aus. Übrig bleibt also nur ein Programm mit ungefähr folgender Ausgabe: * * * * ... * * * * * ... Alles in allem ist das neue Programm nicht mehr sehr eindrucksvoll, aber es hat noch genügend Probleme für den Anfang: • Um die Ausgabe einzurücken, definieren wir eine Hilfs-Funktion „spaces“, die einen StringBuffer der entsprechenden Größe zurückgibt. Die Funktion ist mit ihrer Schleife nicht besonders elegant, aber manchmal muss man so programmieren. • Damit die Ausgabe nicht nur so dahin-fliegt, müssen wir das Programm verlangsamen. Nach jeder Ausgabe legen wir es daher für 30 ms schlafen. Auch hierfür gibt es eine extra Funktion „sleep“, die für uns den „Thread.sleep“ Aufruf kapselt, und den notwendigen try/catch-Block übernimmt. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 213 / 409 • Statt einem festen Programm-Ende oder z.B. einer Tastatur-Abfrage aufs Programm-Ende arbeiten wir mit einer Endlos-Schleife. Das Programm muss daher auf der Konsole mit „Strg+C“ oder in der IDE abgebrochen werden. public class Appl { public static StringBuffer spaces(int len) { StringBuffer sb = new StringBuffer(len); for (int i=0; i<len; i++) { sb.append(' '); } return sb; } public static void sleep() { try { Thread.sleep(30); } catch (Exception x) { } } public static void main(String[] args) { System.out.println("Ein Stern laeuft ueber den Bildschirm"); int pos = 0; while (true) { System.out.println(spaces(pos) + "*"); sleep(); ++pos; if (pos>40) { pos=0; } } } } Das eigentliche Programm reduziert sich so auf eine Initalisierung und eine Endlos-Schleife mit 6 Zeilen: int pos = 0; while (true) { System.out.println(spaces(pos)+"*"); sleep(); ++pos; if (pos>40) { pos=0; } } // (*) // (**) // (***) An welcher Stelle könnte dieses Programm durch eine weitere Abstraktion vereinfacht werden? Immerhin die Hälfte des Programms beschäftigt sich mit dem Positions-Zähler „pos“. Statt eines „int“ Elements wird hier ja ein sogenannter Ring-Zähler benötigt, und der größte Teil der Programmlogik beschäftigt sich jetzt mit dem Problem einen Ring-Zähler zu simulieren. Ausserdem ist nicht schön, dass sich die zwei wichtigsten Parameter des Ring-Zählers an mehreren Stellen wiederfinden: Start- und End-Wert (*), (**) und (***). 11.3.2 Überlegung 1 Damit ist Überlegung 1 „Art der Objekte“ beantwortet: wir wollen einen Ring-Zähler © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 214 / 409 implementieren, d.h. eine Art Integer-Variable die bei Erreichen eines bestimmten Wertes wieder auf einen Startwert springt. Bemerkung – mancher wird nun denken: „Was soll das?“ Die Schleife ist doch übersichtlich, das Problem klar, die Lösung einfach – wozu hier Aufwand investieren? Aber diese Einstellung greift zu kurz, denn: • Nicht immer sind alle Code-Vorkommen eines Ring-Zählers so klar und nah beieinander lokalisiert. Nicht selten finden sich diese Stellen in realen Programmen verteilt in mehreren Stellen wieder. Und in solchen Fällen ist dem Leser nicht immer klar, warum hier solcher Code steht. Wir haben es hier halt mit einem sehr einfachen Beispiel-Programm zu tun. • Und selbst wenn der Code und die Lösung einfach sind: man kann auch einfache Sachen vergessen oder falsch machen. • Ein fertiger Ring-Zähler – wir reden hier jetzt von der zweiten Verwendung der Klasse, nicht von ihrer ersten – ist ein fertiger Ring-Zähler. Er ist getestet und im Einsatz und läuft – es ist ein fertiger Ring-Zähler. Man kann ihn nehmen und benutzen. Das nennt man Wiederverwendung und sollte ein Ziel jeder Software-Entwicklung sein. • Und warum müssen Abstraktions-Ebenen einen großen Abstand haben, und die Einheiten dazwischen kompliziert damit sie sich lohnen? Sind sie groß und kompliziert, so sind sie wahrscheinlich auch fehlerhaft. Sind sie groß und kompliziert, so passen sie sicher nicht beim nächsten Problem. • Und bedenken sie – es ist unser erstes Beispiel. Das ändert nichts daran, dass ein RingZähler als wiederverwendbare Einheit (Klasse) Sinn macht. Aber es gibt sicher auch noch komplexere Klassen. 11.3.3 Überlegung 2 Die Überlegung 2 zielt dann auf die Schnittstelle der Klasse: • Was soll mit den Objekten der Klasse machbar sein? • Wie sollen sich die Objekte verhalten? • Welche Schnittstelle benötigt die Klasse dafür? Hier ist die Lösung einfach: • Bei der Erzeugung des Ring-Zählers müssen Start- und Grenzwert definiert werden können. • Der aktuelle Wert muss abfragbar sein. • Der Wert muss incrementiert werden können, wobei eine Überschreitung des Grenzwerts implizit wieder zum Startwert führen muss. Programm-technisch könnte dies aus der Sicht des Benutzers so aussehen: RingCounter rc = new RingCounter(0, 40); while (true) { System.out.println(spaces(rc.get()) + "*"); sleep(); rc.add(); } 6 Zeilen statt 9, und außerdem noch der Vorteil eine if-Anweisung los geworden zu sein. Und © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 215 / 409 jede Kontrollstruktur bringt halt Komplexität und potentielle Fehlerquellen mit sich. 11.3.4 Überlegung 3 Da wir noch keine Vererbung kennen, ist diese Überlegung zur Zeit noch nicht relevant. 11.3.5 Überlegung 4 So bleibt noch die Frage nach der Implementierung, d.h. wie implementiere ich das? 11.4 Klassen-Implementierung 11.4.1 Definition Zuerst muss eine Klasse definiert werden. Syntax Kassen-Definition [Modifizierer] class klassenname { ... } Achtung – eine Klassen-Definition hat immer ihre eigene Datei, die genauso wie die Klasse mit der Extension ‘java’ heissen muss. Dies ist ein gern gemachter Anfänger-Fehler. Wenn sich ihre Klasse also nicht compilieren lässt - überprüfen sie erstmal die Namen. Die für Klassen möglichen Modifizierer werden später aufgeführt und erklärt. Im Augenblick sollten sie hier immer nur ‘public’ verwenden. Beispiele: public class RingCounter { ... } public class Person { ... } Innerhalb einer Klasse können Konstruktoren, Element-Funktionen, Klassen-Funktionen, Attribute, innere Klassen, und einige weitere Elemente vorhanden sein. Eine Klasse kann aber auch einfach leer sein. 11.4.2 Benutzung Ist eine Klasse definiert, so kann sie im gleichen Package einfach unter ihrem Typ-Namen benutzt werden. Achtung – Variablen der Klasse sind immer Referenz-Variablen. Und erzeugt werden Objekte der Klasse mit dem „new“ Operator. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 216 / 409 public class A { } public class Appl { public static void main(String[] args) { A a = new A(); f(a); a = g(); } public static void f(A a) { } public static A g() { return new A(); } } 11.4.3 Garbage Collector Erzeugte Objekte müssen nicht explizit gelöscht werden. Dies macht die JVM automatisch. Dafür gibt es im System den sogenannten Garbage-Collector, der automatisch im Hintergrund alle nicht mehr referenzierten Objekte löscht. Den Vorgang des Einsammelns nicht mehr benötigter Objekte nennt man Garbage-Collection. Je nach Implementierung in der JVM wird diese von Zeit zu Zeit aufgerufen, oder läuft kontinuierlich im Hintergrund. Außerdem läßt sich die GC auch aus dem Programm heraus mit „System.gc()“ aufrufen – angewendet wird dies in Aufgabe 12.8.1. 11.4.4 Element-Funktionen Leere Klassen sind meist ziemlich sinnlos. Um mit den Objekten der Klasse agieren zu können, sollten diese Element-Funktionen haben. Syntax Funktions-Definition [Modifizierer] rueckgabetyp funktionsname(parameterliste) { // Funktions-Rumpf } Gegenüber den bisher verwendeten Klassen-Funktionen unterscheiden sich ElementFunktionen durch den fehlenden Modifier „static“, und dass sie nur mit Objekt-Bezug aufgerufen werden können. public class A { public static void f() { } public void g() { } public static void main(String[] args) { f(); g(); // Okay, da Klassen-Funktion // Compiler-Fehler, da Element-Funktion -> Objekt-Bezug © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 217 / 409 noetig A a = new A(); a.g(); // Okay, g() wird fuer das Objekt a aufgerufen } } Die für Funktionen möglichen Modifizierer werden später aufgeführt und erklärt. Im Augenblick sollten sie bei Element-Funktionen immer nur ‘public’ verwenden, bei Klassen-Funktionen „public“ und „static“. Und wie sieht unser Ring-Zähler jetzt aus? public class RingCounter { public void add() { ... } // Die Implementierung ist noch offen public int get() { ... } // Die Implementierung ist noch offen } 11.4.5 Warum Element-Funktionen? Weshalb gibt es überhaupt Element-Funktionen? Warum nimmt man nicht einfach KlassenFunktionen, und spart sich den Aufwand mit den Objekten? Man kann beliebig viele Objekte einer Klasse erzeugen kann, die alle voneinander abhängig sind. Element-Funktionen beziehen sich jeweils nur auf das Objekt, für das sie aufgerufen werden, d.h. jedes Objekt kann (und hat) seinen eigenen von allen anderen unabhängigen Status. Klassen-Funktionen beziehen sich auf die Klasse, für die es nur einen Status gibt. Ohne Objekte und Element-Funktionen wäre sowas nicht möglich. StringBuilder s1 = new StringBuilder("Hallo Welt"); StringBuilder s2 = new StringBuilder("Java"); System.out.println("Die Laenge von \"" + s1 + "\" ist " + s1.length()); System.out.println("Die Laenge von \"" + s2 + "\" ist " + s2.length()); „s1“ und „s2“ sind zwei unabhängige Objekte, die unterschiedliche Strings repräsentieren. Nur Funktionen mit Objekt-Bezug können wissen, welches Objekt sie denn nun bearbeiten sollen. Abb. 11-4 : Funktionen ohne bzw. mit Objekt-Bezug 11.4.6 Attribute Natürlich hält ein Objekt Daten. Der Ring-Zähler z.B. muss sowohl seinen Start- und Endwert kennen, als auch seinen aktuellen Wert. Objekt-spezifische Daten – d.h. Variablen in Objekten – werden Attribute genannt. Element-Funktionen können einfach unter Verwendung des Attribut-Namens auf die Attribute zugreifen und sie benutzen. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 218 / 409 Achtung – in Java wird auch gerne der historisch begründetet Name „Feld“ statt „Attribute“ verwandt. Syntax Attribut-Definitionen [Modifizierer] typ name; Beispiel public class A { private int i; private String s; public void set(int arg1, String arg2) { i = arg1; s = arg2; } public void print() { System.out.println(i + " => " + s); } } public class Appl { public static void main(String[] args) { A a1 = new A(); A a2 = new A(); a1.set(1, "Hallo"); a2.set(2, "Java"); a1.print(); a2.print(); } } Ausgabe 1 => Hallo 2 => Java Die für Attribute möglichen Modifizierer werden später aufgeführt und erklärt. Im Augenblick sollten sie bei Attributen immer nur ‘private’ verwenden. Und wie sieht unser Ring-Zähler jetzt aus? public class RingCounter { private int start; private int limit; private int value; public void add() { if (++value>limit) { value=start; } } public int get() { return value; © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 219 / 409 } } Jetzt bleibt nur noch die Frage offen, wie denn die Attribute sinnvoll initalisiert werden, d.h. mit den gewünschten Start- und End-Werten? Zur Zeit werden sie als „int“s alle mit „0“ initialisiert. Hinweis – Attribute werden auch oft Member, Attributes, Element-Variablen, Objekt-Variablen oder (speziell in Java) auch Felder genannt. 11.4.7 Konstruktoren Um ein Objekt bei der Erzeugung sauber zu initialisieren, gibt es spezielle Methoden - die Konstruktoren. Sie haben keinen Rückgabetyp und heissen immer wie die Klasse. Sie werden automatisch immer beim Erzeugen eines neuen Objektes aufgerufen. public class Person { private String forename; private String surename; private int age; public Person(String fn, String sn, int a) { forename = fn; surename = sn; age = a; } public void print() { System.out.println("Name: " + forename + " " + surename); System.out.println("Alter: " + age); } } Person p = new Person("Detlef", "Wilkening", 16); p.print(); // Das Alter stimmt nicht Ausgabe Name: Detlef Wilkening Alter: 16 Nur wenn sie keinen Konstruktor definieren, erzeugt der Compiler automatisch einen sogenannten Standard-Konstruktor, der ohne Argumente aufgerufen werden kann. Der Konstruktor wird entsprechend denen bei new angegebenen Argumenten ausgewählt. public class A { } Person p = new Person(); nicht A a = new A(); Konstruktor // Compiler-Fehler, dieser Konstruktor existiert hier // Klasse A hat einen automatisch erzeugten Standard- Beispiele etwas aufbohren – todo… Und wie sieht unser Ring-Zähler jetzt aus? public class RingCounter { © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 220 / 409 private int start; private int limit; private int value; public RingCounter(int s, int l) { start = s; limit = l; value = s; } public void add() { if (++value>limit) { value=start; } } public int get() { return value; } } Nun ist die Klasse vollständig und unser Beispiel funktioniert. Bemerkung – die Ring-Zähler Klasse ist in diesem Zustand natürlich nicht wirklich in einem produktiven Zustand. Z.B. kann man sie mit einem End-Wert, der kleiner als der Start-Wert ist, problemlos aushebeln. 11.5 Aufgaben 11.5.1 Aufgabe „Ringzähler“ Schreiben Sie einen Ringzähler. Ein Ringzähler ist ein Zähler, der von einem Startwert an hoch (oder runter) zählt (Delta muss nicht 1 sein) und nach Überschreiten des Endwertes wieder beim Startwert anfängt. Ein solcher Zähler wird in der Praxis häufig gebraucht, z.B. als Index in ein Array, als Verweis auf ein Spielfeld, usw. Wie gehen Sie mit widersprüchlichen Eingaben um? Beim Erzeugen eines Ringzählers sollte der Nutzer folgende Möglichkeiten haben: • Angabe nur vom End-Wert => Start-Wert ist 0, Delta ist 1 • Angabe von Start- und End-Wert => Delta ist 1 • Angabe von Start- und End-Wert und Delta Der Ringzähler sollte z.B. mit folgenden Dingen umgehen können – denken Sie sich eine gute Fehlerbehandlung im Rahmen ihrer Möglichkeiten für problematische Fälle aus: • End-Wert vor Start-Wert, Delta positiv • End-Wert nach Start-Wert, Delta negativ • Delta ist 0 Lösung siehe Kapitel 11.6. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 221 / 409 11.5.2 Aufgabe „Kontaktdaten 1“ Schreiben Sie eine kleine Kontaktdaten-Verwaltung. Folgende Fähigkeiten soll das Programm in der ersten Version haben: • Menü für folgende Aufgaben: - Eingabe neuer Personen - Ausgabe aller Personen – unsortiert - Ausgabe aller Personen, deren Vor- oder Nach-Name mit einem bestimmten StringMuster anfängt (Groß- und Kleinschreibung wird nicht unterschieden). • Das Menü soll der Benutzer durch die Eingabe einzelner Buchstaben (mit Return) anwählen können. Personen bestehen aus: • Vorname • Nachname • Telefon (als String) Mögliche Ein- und Ausgabe: Bitte - l : - n : - s : - e : > l 5 - waehlen Sie eine Aktion aus Liste Neu Suchen Ende Kontakte: Detlef Wilkening - Tel: 1234 Dieter Schmidt Dietmar Mueller - Tel: 456 Edgar Wilkens Ralf Eberleh - Tel: 555 444 Bitte - l : - n : - s : - e : > n waehlen Sie eine Aktion aus Liste Neu Suchen Ende Geben Sie eine neue Person ein Vorname: Martina Nachname: Winkens Telefon: 98 76 54 Bitte - l : - n : - s : - e : > l 6 - waehlen Sie eine Aktion aus Liste Neu Suchen Ende Kontakte: Detlef Wilkening - Tel: 1234 Dieter Schmidt Dietmar Mueller - Tel: 456 Edgar Wilkens Ralf Eberleh - Tel: 555 444 Martina Winkens - Tel: 98 76 54 Bitte waehlen Sie eine Aktion aus - l : Liste © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 > Seite 222 / 409 n : Neu s : Suchen e : Ende s Geben Sie einen Suchstring ein: e - Edgar Wilkens - Ralf Eberleh - Tel: 555 444 Bitte - l : - n : - s : - e : > s waehlen Sie eine Aktion aus Liste Neu Suchen Ende Geben Sie einen Suchstring ein: de - Detlef Wilkening - Tel: 1234 Bitte - l : - n : - s : - e : > s waehlen Sie eine Aktion aus Liste Neu Suchen Ende Geben Sie einen Suchstring ein: x - keinen passenden Eintrag gefunden Bitte - l : - n : - s : - e : > e waehlen Sie eine Aktion aus Liste Neu Suchen Ende Ende Schauen Sie sich bei der Lösung ruhig noch mal Aufgabe „Telefonbuch“ (Kapitel 9.8.6) an. Aber Achtung – viele Dinge sind hier anders: • Es gibt ein Menü • Die Liste aller Personen ist hier unsortiert • Personen können eingegeben werden • Das Suchen funktioniert auch mit Anfangs-Mustern ohne Berücksichtigung von Groß- und Klein-Schreibung. • Und vor allem: jetzt kennen sie Klassen. Und da Sie ja wissen, dass wir die Kontaktdaten Verwaltung später noch aufbohren wollen – sollten Sie sie sauber strukturieren. Lösung siehe Kapitel 11.7. 11.5.3 Aufgabe „Kontaktdaten 2“ Und damit wir gleich sehen, ob Sie Ihr Programm sauber strukturiert haben, bohren wir es direkt auf: Eine Person soll neben Vor-, Nachname und Telefon auch noch eine Beschreibung (String) enthalten. Mögliche Ein- und Ausgabe: Bitte waehlen Sie eine Aktion aus © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 > l n s e n : : : : Seite 223 / 409 Liste Neu Suchen Ende Geben Sie eine neue Person ein Vorname: Detlef Nachname: Wilkening Telefon: 1234 Beschreibung: Java Dozent Bitte - l : - n : - s : - e : > n waehlen Sie eine Aktion aus Liste Neu Suchen Ende Geben Sie eine neue Person ein Vorname: Bernd Nachname: Mustermann Telefon: Beschreibung: Bitte - l : - n : - s : - e : > l waehlen Sie eine Aktion aus Liste Neu Suchen Ende 2 Kontakte: - Detlef Wilkening - Tel: 1234 => Java Dozent - Bernd Mustermann Bitte - l : - n : - s : - e : > e waehlen Sie eine Aktion aus Liste Neu Suchen Ende Ende Wenn Sie ihr Programm gut designt haben, dann sollte dies mit sehr wenig Änderungen am Code machbar sein. Und die Änderungen sollten auch nur sehr lokal sein. Lösung siehe Kapitel 11.8. 11.6 Lsg. zu Aufgabe „Ringzähler“ – Kap. 11.5.1 Erklärung folgt (hoffentlich) später – todo... Achtung – ich benutze hier schon das Sprachmittel „this“ aus dem Kapitel 12.4.3. public class RingCounter { private private private private int int int int start; limit; delta; value; public RingCounter(int limit) { © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 224 / 409 this(0, limit, 1); } public RingCounter(int start, int limit) { this(start, limit, 1); } public RingCounter(int start, int limit, int delta) { if (delta==0) { delta = 1; } else if (delta<0) { delta = -delta; } this.start = start; this.limit = limit; this.delta = start<=limit ? delta : -delta; this.value = start; } public int next() { value += delta; if ((value>limit && delta>0) || (value<limit && delta<0)) { value = start; } return value; } public int get() { return value; } } public class Appl { public static void main(String[] args) { System.out.println("RingCounter: 0..4 / 1"); RingCounter rc = new RingCounter(4); System.out.print(rc.get() + " "); for (int i=0; i<20; i++) { System.out.print(rc.next() + " "); } System.out.println('\n'); System.out.println("RingCounter: 2..6 / 1"); rc = new RingCounter(2, 6); System.out.print(rc.get() + " "); for (int i=0; i<20; i++) { System.out.print(rc.next() + " "); } System.out.println('\n'); System.out.println("RingCounter: 2..-2 / 1"); rc = new RingCounter(2, -2); System.out.print(rc.get() + " "); for (int i=0; i<20; i++) { System.out.print(rc.next() + " "); } System.out.println('\n'); System.out.println("RingCounter: 1..9 / 0"); rc = new RingCounter(1, 9, 0); System.out.print(rc.get() + " "); for (int i=0; i<20; i++) { System.out.print(rc.next() + " "); } System.out.println('\n'); System.out.println("RingCounter: 4..40 / 7"); rc = new RingCounter(4, 40, 7); System.out.print(rc.get() + " "); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 225 / 409 for (int i=0; i<20; i++) { System.out.print(rc.next() + " "); } System.out.println('\n'); System.out.println("RingCounter: 20..-5 / 3"); rc = new RingCounter(20, -5, 3); System.out.print(rc.get() + " "); for (int i=0; i<20; i++) { System.out.print(rc.next() + " "); } System.out.println('\n'); System.out.println("RingCounter: 0..100 / -18"); rc = new RingCounter(0, 100, -18); System.out.print(rc.get() + " "); for (int i=0; i<20; i++) { System.out.print(rc.next() + " "); } System.out.println(); } } Ausgabe RingCounter: 0..4 / 1 0 1 2 3 4 0 1 2 3 4 0 1 2 3 4 0 1 2 3 4 0 RingCounter: 2..6 / 1 2 3 4 5 6 2 3 4 5 6 2 3 4 5 6 2 3 4 5 6 2 RingCounter: 2..-2 / 1 2 1 0 -1 -2 2 1 0 -1 -2 2 1 0 -1 -2 2 1 0 -1 -2 2 RingCounter: 1..9 / 0 1 2 3 4 5 6 7 8 9 1 2 3 4 5 6 7 8 9 1 2 3 RingCounter: 4..40 / 7 4 11 18 25 32 39 4 11 18 25 32 39 4 11 18 25 32 39 4 11 18 RingCounter: 20..-5 / 3 20 17 14 11 8 5 2 -1 -4 20 17 14 11 8 5 2 -1 -4 20 17 14 RingCounter: 0..100 / -18 0 18 36 54 72 90 0 18 36 54 72 90 0 18 36 54 72 90 0 18 36 11.7 Lsg. zu Aufgabe „Kontaktdaten 1“ – Kap. 11.5.2 Erklärung folgt (hoffentlich) später – todo... Achtung – ich benutze hier schon das Sprachmittel „Klassen-Funktionen“ in einer Form, wie wir sie hier noch nicht kennen (siehe Kapitel 12.4.4) und auch die direkte Attribut-Initialisierung, wie sie erst in Kapitel 12.2.1 vorgestellt wird. import java.io.BufferedReader; import java.io.InputStreamReader; public class Keyboard { private static InputStreamReader isr = new InputStreamReader(System.in); private static BufferedReader reader = new BufferedReader(isr); public static String readString() { try { return reader.readLine(); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 226 / 409 } catch (Exception x) { } return ""; } } public class Person { private String forename = ""; private String surename = ""; private String phone = ""; public boolean find(String pattern) { int len = pattern.length(); if (forename.length()>=len) { String begin = forename.substring(0, len); if (begin.equalsIgnoreCase(pattern)) { return true; } } if (surename.length()>=len) { String begin = surename.substring(0, len); if (begin.equalsIgnoreCase(pattern)) { return true; } } return false; } public void input() { System.out.print("Vorname: "); forename = Keyboard.readString(); System.out.print("Nachname: "); surename = Keyboard.readString(); System.out.print("Telefon: "); phone = Keyboard.readString(); } public void print() { System.out.print("- " + forename + " " + surename); if (phone.length()!=0) { System.out.print(" - Tel: " + phone); } System.out.println(); } } import java.util.ArrayList; import java.util.Iterator; public class Contacts { private ArrayList<Person> contacts = new ArrayList<>(); public int size() { return contacts.size(); } public void add(Person p) { contacts.add(p); } public int find(String pattern) { int count = 0; Iterator<Person> i = contacts.iterator(); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 227 / 409 while (i.hasNext()) { Person p = i.next(); if (p.find(pattern)) { p.print(); count++; } } return count; } public void print() { Iterator<Person> i = contacts.iterator(); while (i.hasNext()) { i.next().print(); } } } public class Appl { private Contacts cts = new Contacts(); public void run() { while (true) { System.out.println("Bitte waehlen Sie eine Aktion aus"); System.out.println("- l : Liste"); System.out.println("- n : Neu"); System.out.println("- s : Suchen"); System.out.println("- e : Ende"); System.out.print("> "); String in = Keyboard.readString(); System.out.println(); if (in.equalsIgnoreCase("l")) { print(); } else if (in.equalsIgnoreCase("n")) { insert(); } else if (in.equalsIgnoreCase("s")) { find(); } else if (in.equalsIgnoreCase("e")) { System.out.println("Ende"); return; } System.out.println(); } } private void print() { System.out.println(cts.size() + " Kontakte:"); cts.print(); } private void insert() { System.out.println("Geben Sie eine neue Person ein"); Person p = new Person(); p.input(); cts.add(p); } private void find() { System.out.print("Geben Sie einen Suchstring ein: "); String pattern = Keyboard.readString(); int count = cts.find(pattern); if (count==0) { System.out.println("- keinen passenden Eintrag gefunden"); } } public static void main(String[] args) { Appl appl = new Appl(); appl.run(); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 228 / 409 } } 11.8 Lsg. zu Aufgabe „Kontaktdaten 2“ – Kap. 11.5.3 Die neue Kontaktdaten-Verwaltung unterscheidet sich durch die alte nur durch ein weiters Merkmal der zu verwaltenen Personen. Da alle personen-relevanten Elemente in der PersonenKlasse gekapselt sind, ist nur diese Klasse betroffen. Das komplette restliche Programm ist von der Änderung nicht betroffen. Daher wird hier auch nur die neue Personen-Klasse besprochen und gelistet. Was muß an der Personen-Klasse geändert werden? • Sie benötigt ein weiteres String-Attribut für die Beschreibung. • Die Beschreibung muss bei der Personen-Eingabe mit von der Tastatur gelesen werden – siehe Element-Funktion „input“. • Und die Beschreibung muß – wenn vorhanden – mit ausgegeben werden – siehe ElementFunktion „print“. • Das Suchen wird von der Beschreibung nicht beeinflußt – also muß die Element-Funktion „find“ nicht angepaßt zu werden. Fertig – das war’s. public class Person { private private private private String String String String forename = ""; surename = ""; phone = ""; description = ""; public boolean find(String pattern) { int len = pattern.length(); if (forename.length()>=len) { String begin = forename.substring(0, len); if (begin.equalsIgnoreCase(pattern)) { return true; } } if (surename.length()>=len) { String begin = surename.substring(0, len); if (begin.equalsIgnoreCase(pattern)) { return true; } } return false; } public void input() { System.out.print("Vorname: "); forename = Keyboard.readString(); System.out.print("Nachname: "); surename = Keyboard.readString(); System.out.print("Telefon: "); phone = Keyboard.readString(); System.out.print("Beschreibung: "); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 229 / 409 description = Keyboard.readString(); } public void print() { System.out.print("- " + forename + " " + surename); if (phone.length()!=0) { System.out.print(" - Tel: " + phone); } if (description.length()!=0) { System.out.print(" => " + description); } System.out.println(); } } 12 Klassen-Details 12.1 Zugriffsbereiche Java kennt 4 Zugriffs-Modifizierer. Für alle Elemente in einer Klasse haben sie folgende Bedeutung: • public Auf diese Elemente darf jeder von überall her zugreifen. • protected Auf diese Elemente darf von innerhalb des gleichen Package und von den abgeleiteten Klassen her zugegriffen werden. Packages werden in Kapitel 13 eingeführt, Vererbung in Kapitel 14. D.h. ist dieser Modifizierer für uns zur Zeit ohne Bedeutung. • --- (kein Zugriffs-Modifizierer entspricht einer Package Sichtbarkeit) Auf diese Elemente darf von innerhalb des gleichen Package her zugegriffen werden. Packages werden in Kapitel 13 eingeführt. D.h. ist dieser Modifizierer für uns zur Zeit noch ohne Bedeutung. • private Auf diese Elemente darf nur von innerhalb der Klasse zugegriffen werden (und von inneren Klassen – siehe Kapitel 15). public class A { private void f(); public void g() { f(); } // Kein Problem, da innerhalb der Klasse } public class B { public void f(A a) { a.f(); a.g(); } // Compiler-Fehler, da private // Okay, da public } Wichtige Bemerkung – es ist sehr sehr sehr empfehlenswert alle Attribute private zu machen, © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 230 / 409 und einen Zugriff von außen – wenn er denn überhaupt notwendig ist – über Funktionen zu regeln. Dies ist einer der wichtigsten Grundprinzipien jeglicher Software-Entwicklung und der Objektorientierung: „Information-Hiding“: • Die interne Implementierung ist vollständig vor dem Benutzer versteckt. • Die interne Implementierung läßt sich jederzeit ändern. • Von außen können keine inkonsistenten Objekt-Zustände erzeugt werden. 12.2 Konstruktoren 12.2.1 Initialisierung Wird ein Objekt erzeugt, so gibt es mehrere Möglichkeiten die Attribute zu initialisieren. • Man macht nichts: in diesem Fall werden die entsprechenden Attribute entsprechend ihrem Typ initialisiert. • Man kann den gewünschten Wert direkt bei der Attribut-Definition angeben. • Eine Klasse kann Initialisierungs-Blöcke enthalten – dies wird in der Vorlesung nicht besprochen. • Oder das Attribut wird im Konstruktor initialisiert. public class A { private int i1; private int i2 = 42; private int i3; Ko. private private fuer s2 private private Ko. // Default-Wert 0 // Initial-Wert bei der Attribut-Definition // Sieht nach Default-Wert aus, Initialisierung dann im String s1; String s2 = "Hallo"; // Default-Wert null // Initial-Wert String "Hallo" String s3 = new String("Hallo"); String s4; // Langfassung von s2 // Sieht nach null aus, aber dann public A(int arg1, String arg2) { i3 = arg1; s4 = arg2; } } • Direkte Initialisierungen der Attribute ist vor allem dann sinnvoll, wenn eine Klasse mehrere Konstuktoren hat, und die Attribute in allen Konstruktoren die gleichen Werte bekommen. • Das Initialisieren der Attribute in den Konstruktoren ist vor allem dann sinnvoll, wenn die gesetzten Werte konstruktor-spezifisch sind, d.h. z.B. von den Konstruktor-Parametern abhängen. • Das Initialisieren der Attribute in den Initialisierungs-Blöcken – was hier nicht besprochen wird – ist vor allem dann sinnvoll, wenn das Setzen in allen Konstruktoren identisch, aber zu komplex für eine direkte Initialisierung ist. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 231 / 409 12.2.2 Konstruktor-Verkettung Wenn eine Klasse mehrere Konstruktoren hat, so können diese oft aufeinander zurückgeführt werden, um das Programmieren zu vereinfachen. public class Person { private String forename; private String surname; private int age; public Person(String fn, String sn) { forename = fn; surname = sn; } public Person(String fn, String sn, int a) { this(fn, sn); age = a; } } Der Aufruf eines anderen Konstruktors der gleichen Klasse geschieht über den „this“ Zeiger mit den entsprechenden Parametern und muss in der ersten Zeile des Konstruktors stehen. 12.3 finalize Analog zu den Konstruktoren gibt es in Java auch eine Klassen-Methode, die beim Zerstören eines Objektes aufgerufen wird - sie heisst finalize public class A { protected void finalize() { } } Achtung – es ist vollkommen undefiniert wann die Funktion „finalize“ aufgerufen wird. Es kann sogar passieren, dass sie gar nicht aufgerufen wird: • Zwischen „nicht-mehr-referenziert-werden“ und dem Ende des Programms wird keine GC mehr ausgelöst. • Die Verkettung in der Vererbungs-Hierarchie ist unterbrochen – siehe Kapitel todo. U.a. deshalb wird in der Praxis von der Verwendung der Finalize-Funktion abgeraten. Statt dessen sollen lieber Weak- oder Phantom-Referenzen benutzt werden – die zu besprechen sprengt aber mal wieder den Zeit-Rahmen der Vorlesung. Mit einem kleinen Beispiel-Programm kann man den Aufruf der Funktion gut nachverfolgen: public class Test { public Test() { System.out.println("- Konstruktor"); } protected void finalize() { System.out.println("- finalize"); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 232 / 409 } } System.out.println("Objekt wird erzeugt"); Test t = new Test(); System.out.println("Referenz auf das Objekt wird freigeben"); t = null; System.out.println("GC wird ausgeloest"); System.gc(); System.out.println("GC wird ausgeloest"); System.gc(); System.out.println("GC wird ausgeloest"); Mögliche Ausgabe Objekt wird erzeugt - Konstruktor Referenz auf das Objekt wird freigeben GC wird ausgeloest - finalize GC wird ausgeloest GC wird ausgeloest Die Ausgabe kann bei jeder virtuellen Java-Maschine anders aussehen, da das genaue Verhalten bei einer GC der JVM freigestellt ist. Manche Implementierungen z.B. geben ein nicht mehr referenziertes Objekt erst nach der zweiten GC frei, während sie die erste GC nur als Markierungsphase benutzen. 12.4 Funktionen 12.4.1 Parameter und Rückgaben Die integrierten Datentypen wie int, double, char,... werden per Wert übergeben („call-by-value“ – cbv) – sie stellen also eine Kopie da - Änderungen in der Funktion verändern den Wert in der aufrufenden Funktion nicht. Bei allen anderen Parametern, die ja alle Referenz-Typen sind, werden die Argumente per Referenz übergeben („call-by-reference“ – cbr). Bei diesen referenzieren die Parameter das gleiche Objekt wie in der aufrufenden Funktion – d.h. es kann in der Funktion das OriginalObjekt verändert werden. public class A { private static void f(int i, StringBuilder s) { System.out.println("> f(" + i + ", " + s + ")"); i = 12; s.append(" Welt"); System.out.println("> i: " + i); System.out.println("> s: " + s); } public static void main(String[] args) { int i = 42; StringBuilder s = new StringBuilder("Hallo"); f(i, s); System.out.println("# i: " + i); System.out.println("# s: " + s); } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 233 / 409 } Ausgabe > f(42, Hallo) > i: 12 > s: Hallo Welt # i: 42 # s: Hallo Welt Der Rückgabe-Typ void bei Funktionen gibt an, dass die Funktion nichts zurückgibt. 12.4.2 return Ist eine Funktion nicht vom Typ void, so muss sie einen zu dem Typ passenden Wert zurückgeben. Dies geschieht mit der return Anweisung. Die Funktion wird instantan beendet. public class Person { private int age; public int getAge() { return alter; } } • In einer Funktion können beliebig viele return Anweisungen vorkommen. • Der Rückgabewert kann beim Funktions-Aufruf ignoriert werden, d.h. er muss nicht verwendet oder ausgewertet werden. 12.4.3 this Innerhalb jeder Element-Funktion ist automatisch das Schlüsselwort „this“ definiert, das eine Referenz auf das aktuelle Objekt ist, d.h. das für das die Element-Funktion aufgerufen wurde. Eine Referenz auf das aktuelle Objekt – aus Sicht der Element-Funktion auf sich selber – wird in der Praxis häufig benötigt, z.B. um „sich-selber“ an andere Element-Funktionen zu übergeben. Aber auch für profanere Aufgaben wie z.B. zur Aufruf-Verkettung (siehe Beispiel) wird „this“ eingesetzt. public class A { public A myself() { return this; } public void f() { } } A a = new A(); a.myself().f(); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 234 / 409 Ein ganz typischer Anwendungsfall in Java für „this“ ist die Verwendung gleicher Namen für Attribute (d.h. Variablen im Objekt) und Parameter. In einem solchen Fall verdecken die Parameter die Attribute, da sie im Scope der Funktion liegen, während die Attribute zum Scope der Klasse gehören. Über das Schlüsselwort „this“ – d.h. über das aktuelle Objekt – können die Attribute trotz gleich-namiger lokaler Variablen angesprochen werden. public class RingCounter { private private private private int int int int start; limit; delta; value; public RingCounter(int start, int limit, int delta) { this.start = start; this.limit = limit; this.delta = delta; this.value = start; } } Achtung – in Klassen-Funktionen, d.h. Funktionen mit dem Modifierer static, ist „this“ nicht definiert, da Klassen-Funktionen keinen Objekt-Bezug haben. 12.4.4 Klassen-Funktionen Klassen-Funktionen – d.h. Funktionen mit dem Modifier „static“ – werden ohne Objekt-Bezug aufgerufen. Klassen-Funktionen von fremden Klassen müssen mit dem Klassen-Namen referenziert werden. public class A { public static void f() { System.out.println("A.f"); } } public class B { public static void f() { System.out.println("B.f"); } public static void main(String[] args) { f(); B.f(); A.f(); } } Ausgabe B.f B.f A.f Im Gegensatz zu einer Element-Funktion ist eine Klassen-Funktion nicht einem Objekt, sondern einer Klasse zugeordnet. Daher können Klassen-Funktionen natürlich nicht auf Attribute zugreifen, sondern nur auf Klassen-Variablen – siehe Kapitel 12.5.1. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 235 / 409 12.5 Attribute 12.5.1 Klassen-Variablen So wie es Element- und Klassen-Funktionen gibt, so gibt es auch Element- und KlassenVariablen. Gegenüber Attributen haben Klassen-Variablen zusätzlich den Modifier „static“. public class A { private static int i; private static String s = "Hallo"; } Im Gegensatz zu einem Attribut ist eine Klassen-Variable nicht einem Objekt, sondern einer Klasse zugeordnet. Klassen-Variablen können von Element- und von Klassen-Funktionen genutzt werden. Klassen-Variablen werden initialisiert, wenn die Klasse in die JVM geladen wird. Für die Definition der Initialisierung gibt es drei Arten: • Man macht nichts: in diesem Fall werden die entsprechenden Klassen-Variablen entsprechend ihrem Typ initialisiert. • Man kann den gewünschten Wert direkt bei der Klassen-Variablen -Definition angeben. • Eine Klasse kann statische Initialisierungs-Blöcke enthalten – dies wird in der Vorlesung nicht besprochen. 12.5.2 Klassen-Konstanten An vielen Stellen sind beim Programmieren Konstanten – gerade Integer Konstanten – sehr hilfreich. Eine Konstante ermöglicht es ihnen statt eines Literals50 einen sprechenden Namen zu benutzen, und den Wert der Konstanten an einer einzigen Stelle zu definieren, egal wie oft und wo sie die Konstante benutzen. // Bsp ohne Konstante double bruttoPrice = 1.19 * nettoPrice; // (*) ... System.out.println(value/1.19); // (**) // Bsp mit Konstante – Achtung, nur Pseudocode MWST = 1.19 double bruttoPrice = MWST * nettoPrice; ... System.out.println(value/ MWST); 50 Literal – siehe auch Kapitel todo. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 236 / 409 Während in Zeile (*) noch recht offensichtlich ist, dass das Literal wohl die Mehrwert- oder Umsatz-Steuer meint, ist dies in Zeile (**) nur noch durch den Kontext erkennbar. Statt dessen ist dies im Beispiel mit Konstante in beiden Fällen sofort klar. Ändert sich die Umsatz-Steuer, so müssen Sie im ersten Beispiel alle Stellen im Programmcode finden, die ein Literal enthalten, und sie ändern. Und hoffentlich vergessen sie keine Stelle bzw. ändern keine „19“ um, die nichts mit der Umsatz-Steuer zutun hat. Im zweiten Beispiel müssen Sie nur die eine Konstanten-Definition ändern, was viel weniger Arbeit ist und auch weniger fehlerträchtig. Ein Konstante ist in Java eine normale Element- oder Klassen-Variable, die den Modifier „final“ bekommt. Damit läßt sich die entsprechende Variable nicht mehr ändern – siehe auch Kapitel 8.3.1. public class A { public static final int CONST_VAR = 42; } Bemerkung – Konstanten sind in Java typischerweise Klassen-Konstanten, d.h. konstante Klassen-Variablen, da die Konstante normalerweise ja für alle Objekte gleich ist. Also mit Modifier „static“. Bemerkung – Klassen-Konstanten sind, obwohl Variablen, häufig „public“ oder „protected“, da sie meist von überall genutzt werden können sollen. Da sie konstant sind, d.h. nicht geändert werden können, ist der unbeschränkte Zugriff ja auch kein Problem. Bemerkung – der Modifier „final“ wirkt bei Referenz-Variablen nur auf die Variable, und nicht auf das referenzierte Objekt. Bemerkung – Klassen-Konstanten werden in Java per Konvention in Großbuchstaben geschrieben, und die einzelnen Wörter durch Underscores getrennt – siehe auch Kapitel 3.2.5. 12.6 Aufzählungen Aufzählungen werden benötigt, wenn eine Variable nur wenige feste Werte annehmen kann, die alle zur Entwicklungs-Zeit schon feststehen. Beispiele hierfür sind: • Ausrichtung von Text in einem Absatz: - Linksbündig, zentriert, rechtsbündig, blocksatz • Spielfarben in einem Kartenspiel: - Karo, Herz, Pik, Kreuz • Belegung eines Feldes beim Tic-Tac-Toe: - Unbelegt, weiß, schwarz • Auflistung der 16 HTML Grund-Farben: © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 - Seite 237 / 409 Schwarz, weiß, rot, grün,… In einem solchen Fall wünscht man sich als Programmierer einen speziellen Typ, der nur die jeweiligen festen Werte aufnehmen kann – eine Aufzählung oder Enumeration, die es seit dem JDK 1.5 in Java gibt. • Intern werden Enums in Java mit Klassen definiert, d.h. Enums sind eigentlich nur Klassen. Daher bestimmt wie bei Klassen auch der Enum-Name den Datei-Namen, und in der Datei steht nur die Enumeration. • Die Enumeration wird mit dem Schlüsselwort „enum“ eingeleitet, und ist meistens „public“. • Im einfachsten Fall besteht der Enum nur aus den Aufzählungs-Konstanten, die vergleichbar zu Klassen-Konstanten sind. Als solche werden sie natürlich groß geschrieben. • Der Enum (im Beispiel „Color“) kann wie eine Klasse genutzt werden – d.h. z.B. für Variablen-Definitionen oder in Funktions-Schnittstellen – natürlich mit Referenz-Semantik. • Java sorgt dafür, dass nur die Aufzählungs-Konstanten in der Enum-Definition existieren. Man kann im Programm keine weiteren Aufzählungs-Konstanten anlegen. Daher funktioniert der Vergleich mit „==“ (Identität), und man muß nicht „equals“ benutzen. public enum Color { RED, GREEN, BLUE } public class A { public static void checkColor(Color c) { if (c == Color.RED) { System.out.println("Farbe ist rot"); } else { System.out.println("Farbe ist nicht rot"); } } public static void main(String[] args) { Color co = Color.GREEN; checkColor(co); checkColor(Color.RED); } } Ausgabe Farbe ist nicht rot Farbe ist rot Ein typisches Einsatzgebiet von Enums sind Switch-Anweisungen, in denen sie genutzt werden können (siehe auch Kapitel 7.2). Color c = Color.GREEN; switch (c) { case RED: System.out.println("Farbe ist rot"); break; case GREEN: System.out.println("Farbe ist gruen"); break; case BLUE: System.out.println("Farbe ist blau"); break; } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 238 / 409 Ausgabe Farbe ist gruen 12.6.1 Enums in Klassen Selbst wenn wir innere Klassen noch nicht kennen (siehe Kapitel 15) – Enums können auch innerhalb von Klassen als eine Art klassenbezogene Aufzählung definiert werden. Das Schlüsselwort „static“ ist hierbei optional. public class A { public static enum Color1 { RED, GREEN, BLUE }; public enum Color2 { RED, GREEN, BLUE }; } 12.6.2 Enums sind Klassen Enums sind in Wirklichkeit echte Klassen, und können als solche auch definiert und genutzt werden. Daher man kann die Enum-Konstanten mit weiteren Informationen (Attributen) und Element-Funktionen anreichern: public enum Color { RED(255,0,0), GREEN(0,255,0), BLUE(0,0,255); private int r,g,b; private Color(int r, int g, int b) { this.r = r; this.g = g; this.b = b; } public int getRedValue() { return r; } } Achtung – Konstruktoren von Enumerationen müssen „private“ sein, da nur die Klasse die Enumerations-Konstanten erzeugen darf. 12.6.3 Weiteres • Im JDK wurden außerdem noch statische Imports eingeführt, die man u.a. mit Enums nutzen kann. Aus Zeitmangel werden wir keine statischen Imports besprechen. • Enums bringen von sich aus einige Funktionen mit sich, wie z.B. die Klassen-Funktionen: - „values“, die ein Array aller Enum-Konstanten zurückgibt. - „valueOf“, die die Enum Konstante zum übergebenen String zurückgibt. for (Color c : Color.values()) { System.out.println(c); } Color c = Color.valueOf("RED"); System.out.println("-> " + c); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 239 / 409 Ausgabe RED GREEN BLUE -> RED 12.6.4 Historie Erst mit Java 5.0 (JDK 1.5) sind Aufzählungs-Typen in Java eingeführt worden. Bis dahin wurden meist Integer-Konstanten für Aufzählungen verwandt, typischerweise so: public class Text { public public public public static static static static final final final final int int int int LEFT = 1; CENTER = 2; RIGHT = 3; BLOCK = 4; public void setAlignment(int arg) { ... } } Obwohl diese Lösung mit Integer-Konstanten alles andere als gut ist (fehlende Typsicherheit), sollten Sie sie kennen, denn sie findet sich in vielen altem Code und vor allem auch in den Schnittstellen der Java Bibliothek, denn diese wurde zum größten Teil schon in den JDKs 1.0 bis 1.4 definiert. Eine etwas bessere Lösung für Aufzählungs-Typen statt Integern ist ihre Simulation über spezielle Klassen. Dieses Verfahren ist in vielen Büchern51 und Artikeln (auch bei Oracle selber) beschrieben – sprengt aber leider den Rahmen des Tutorials. Seit dem JDK 1.5 ist dies eh nicht mehr notwendig, da jetzt ja echte Aufzählungs-Typen in Java zur Verfügung stehen. 12.7 Top-Level-Klassen Klassen (und auch Interfaces), die direkt in ein Package eingebettet sind, nennt man auch TopLevel-Klassen (bzw. Top-Level-Interfaces). Von daher sind alle Klassen, die wir bisher kenenn gelernt haben Top-Level-Klassen. 12.8 Aufgaben 12.8.1 Aufgabe „Türme von Hanoi“ Simulieren Sie die „Türme von Hanoi“. 51 Z.B. in „Effektiv Java“ von todo. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 240 / 409 Abb. 12-1 : Türme von Hanoi Die „Türme von Hanoi“ ist ein Spiel, bei dem am Anfang alle Scheiben auf einem Stab liegen, und dann zu einem anderen Stab transportiert werden sollen. Für den Transport gelten zwei Regeln: • Es darf immer nur eine Scheibe bewegt werden. • Es darf immer nur eine kleinere Scheibe auf einer größeren liegen, nie umgekehrt. Schreiben Sie ein Programm, dass am Anfang die Anzahl an Scheiben einliest, mit denen die Türme von Hanoi gespielt werden sollen. Danach soll das Spiel mit entsprechender Ausgabe simuliert werden. Mögliche Ein- und Ausgabe Anzahl Scheiben: 3 1. -|| | --|-| | ---|--| | ************************* 2. | | | --|-| | ---|--| -|************************* 3. | | | | | | ---|--- --|--|************************* 4. | | | | -|| ---|--- --|-| ************************* 5. | | | | -|| | --|-- ---|--************************* 6. | | -|- | | --|-- | | ---|--- © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 241 / 409 ************************* 7. | | | | | --|--|| ---|--************************* 8. | | -|| | --|-| | ---|--************************* Lösung siehe Kapitel 12.9. 12.8.2 Aufgabe „Gerüchteküche“ Kleine Dörfer sind Gerüchteküchen erster Klasse. Jeder kennt jeden, und jeder redet über jeden, bzw. schweigt wissend. All das wollen wir in einem kleinen Programm nachbilden. Simulieren Sie ein Dorf mit n Einwohnern. Am Anfang kennt nur ein beliebiger Einwohner (nehmen Sie einfach den ersten) das Gerücht. Danach simulieren Sie das Geschehen, indem Sie maximal m Durchläufe starten. In jedem Durchlauf treffen sich zwei zufällige Einwohner des Dorfes und reden miteinander. Was hierbei passiert, ist davon abhängig, welchen Status bzgl. des Gerüchts die beiden Einwohner haben. Einwohner 1 unwissend unwissend unwissend erzählend erzählend erzählend schweigsam schweigsam schweigsam Einwohner 2 unwissend erzählend schweigsam unwissend erzählend schweigsam unwissend erzählend schweigsam => Einwohner 1 unwissend erzählend unwissend erzählend schweigsam schweigsam schweigsam schweigsam schweigsam Einwohner 2 unwissend erzählend schweigsam erzählend schweigsam schweigsam unwissend schweigsam schweigsam Änderung + + + + + - Das Programm ist damit im Prinzip beschrieben. Hier noch einige Hinweise: • Der Benutzer soll am Anfang eingeben, wieviele Bewohner das Dorf hat, und wieviele Durchläufe maximal gewünscht sind. • Gibt es stabile Zustände? D.h. Zustände, bei denen es keine Änderung der Stati der Einwohner mehr gibt? Wenn ja, welche? Erreicht das Programm einen solchen stabilen Zustand, soll die Simulation mit einer entsprechenden Meldung beendet werden. • Wird das Programm durch die max. Anzahl an Durchläufen beendet, soll die Simulation auch hier mit einer entsprechenden Meldung beendet werden. • Das Programm soll nach einer Status-Änderung folgende Informationen in einer Zeile © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 242 / 409 ausgeben52: - Nummer des Durchlaufs (n-stellig in Abhängigkeit der max. Durchläufe, rechtsbündig, führende Leerzeichen) - Anzahl an unwissenden Einwohnern in der Form „U(_x)“ (n-stellig in Abhängigkeit der max. Durchläufe, rechtsbündig, führende Underscores) - Anzahl an erzählenden Einwohnern – Form anlog zu den Unwissenden - Anzahl an schweigsamen Einwohnern – Form anlog zu den Unwissenden - Ein Zeichen für jeden Einwohner: o Unwissend => „_“ o Erzählend => „|“ o Schweigsam => „*“ Eine Simulation könnte also folgendermaßen aussehen: Mögliche Ein- und Ausgabe: Geruechtekueche - Anzahl Personen: 30 - Max Durchlaeufe: 400 0: 3: 5: 6: 8: 9: 19: 26: 36: 41: 45: 47: 62: 109: 116: 136: 139: 159: 182: U(29) U(28) U(27) U(26) U(26) U(25) U(25) U(24) U(23) U(22) U(22) U(22) U(21) U(21) U(21) U(20) U(19) U(19) U(19) E( E( E( E( E( E( E( E( E( E( E( E( E( E( E( E( E( E( E( 1) 2) 3) 4) 2) 3) 2) 3) 4) 5) 4) 2) 3) 2) 1) 2) 3) 1) 0) S( 0) S( 0) S( 0) S( 0) S( 2) S( 2) S( 3) S( 3) S( 3) S( 3) S( 4) S( 6) S( 6) S( 7) S( 8) S( 8) S( 8) S(10) S(11) |_____________________________ |___________________|_________ |___________________|__|______ |_________|_________|__|______ |_________*_________|__*______ |_____|___*_________|__*______ |_____*___*_________|__*______ |_____*___*__|______|__*______ |_____*__|*__|______|__*______ |__|__*__|*__|______|__*______ *__|__*__|*__|______|__*______ *__|__*__|*__*______*__*______ *__|__*__|*__*______*__*___|__ *__|__*__**__*______*__*___|__ *__|__*__**__*______*__*___*__ *__|__*__**__*____|_*__*___*__ *__|__*__**__*___||_*__*___*__ *__|__*__**__*___**_*__*___*__ *__*__*__**__*___**_*__*___*__ Ende wegen stabilem Zustand Lösung siehe Kapitel 12.10. 12.8.3 Aufgabe „Tic-Tac-Toe 1“ Schreiben Sie ein einfaches Tic-Tac-Toe. Tic-Tac-Toe ist ein einfaches Spiel, das auf einem 3x3 Brett gespielt wird. Es wird abwechselnd gezogen – jedesmal legt ein Spieler einen seiner Spielsteine auf ein Brett. Gewonnen hat, wer zuerst 3 eigene Steine in einer Reihe hat (vertikal, horinzontal bzw. diogonal). Nach spätestens neun Zügen ist das Spiel vorbei – hat dann keiner der Spieler eine vollständige Reihe geschafft, ist das Spiel unentschieden. Hinweis – wird das Spiel von einem Spieler optimal gespielt, kann er nicht verlieren. D.h. 52 Achtung – das Programm ändert nicht nach jedem Durchlauf seinen Status. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 243 / 409 spielen beide Spieler optimal, so wird es immer unentschieden ausgehen. Fangen Sie das Programm einfach und klein an, und gehen Sie im ersten Schritt sehr pragmatisch vor. D.h. versuchen Sie das Programm erstmal vollständig (im Sinne von „man kann spielen“) zum Laufen zu bringen: • Machen Sie eine einfache pragmatische Eingabe, z.B. indem die Felder von 1..9 durchnummeriert sind, und Sie einfach nur die Feld-Nummer von der Tastatur abholen. • Machen Sie nach jedem Zug eine einfache Ausgabe in die Konsole, z.B. indem ein Feld mit „_“ (unbelegt), „X“ (Spieler 1) und „O“ (Spieler 2) ausgegeben wird. • Legen Sie einfach fest, dass z.B. immer der Mensch anfängt und der Computer der zweite Spieler ist. • Während des Spiels ist kein Abbruch möglich. • Und vor allen Dingen verwenden Sie erstmal nicht zuviel Arbeit auf einen guten Computerspieler (zum einen würden Sie dann nicht mehr gewinnen können, zum anderen ist dies eher ein algorithmisches und nicht ein Java Problem). Im einfachsten Fall programmieren Sie einfach einen Computerspieler, der das erste leere Feld besetzt. Wenn dann irgendwann alles gut, stabil, schön usw. läuft, dann sollten Sie sich aber ruhig noch an dieses Problem heranwagen und einen optimalen Computerspieler implementieren. Mögliche Ein- und Ausgabe: ___ ___ ___ Bitte machen sie ihren Zug (1..9): 1 Mensch zieht 1 - Feld 1/1 O__ ___ ___ Computer (naechstes freies Feld) zieht 2 - Feld 1/2 OX_ ___ ___ Bitte machen sie ihren Zug (1..9): 4 Mensch zieht 4 - Feld 2/1 OX_ O__ ___ Computer (naechstes freies Feld) zieht 3 - Feld 1/3 OXX O__ ___ Bitte machen sie ihren Zug (1..9): 7 Mensch zieht 7 - Feld 3/1 OXX O__ O__ © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 244 / 409 Mensch hat gewonnen Hinweise: • Lassen Sie sich nicht von dem scheinbar großen Problem einschüchtern. Programmieren heißt immer auch – egal ob prozedural, funktional, objekt-orientiert,.. – ein Problem in kleinere Teilprobleme zu zerlegen, bis man diese lösen kann. • Fangen Sie auch nicht einfach blind an zu Implementieren – dazu ist diese Aufgabe wahrscheinlich schon zu groß. Setzen Sie sich hin, und überlegen Sie, welche Teile Sie zuerst warum und wie implementieren. Welche hängen dann davon ab? • Identifzieren Sie die wichtigsten Konzepte des Spiels (welche sind das?) und packen Sie diese in Klassen. • Überlegen Sie, welche Schnittstellen die Klassen benötigen. Welche Klasse hat welche Aufgaben bzw. Verantwortlichkeiten? • Es ist eine interessante und wichtige Aufgabe sich zu überlegen, an welcher Stelle welche Arbeit erledigt wird – die Kriterien dafür sind Klarheit, Erweiterbarkeit, Allgemeinheit, Wiederverwendbarkeit, usw. Lösung siehe Kapitel 12.11. 12.9 Lsg. zu Aufgabe „Türme von Hanoi“ – Kap. 12.8.1 Erklärung folgt (hoffentlich) später – todo... import java.util.ArrayList; public class Hanoi { private ArrayList startTower, middleTower, endTower; private int count; private int move; //-------------------------------------------------------------------------public void play(int count) { this.count = count; startTower = new ArrayList(count); middleTower = new ArrayList(); endTower = new ArrayList(); for (int i=count; i>0; --i) { startTower.add(i); } move = 1; print(); moveTower(0, startTower, endTower, middleTower); } //-------------------------------------------------------------------------private void moveTower( int layer, ArrayList source, ArrayList dest, ArrayList help) { // Wenn die zu bewegende Ebene die oberste ist, // dann kann die Scheibe direkt bewegt werden. if (layer==source.size()-1) { moveSlice(source, dest); return; © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 245 / 409 } // Um den gesamten Haufen von Scheiben ab der angegebenen Ebene // zu bewegen, muss: // - zuerst der Haufen Scheiben ueber der Ebene zum Hilfs-Turm // bewegt werden, // - dann die Scheibe zum Ziel-Turm bewegt werden, und dann // - der verschobene Haufen auf die bewegte Scheibe bewegt werden. int helpLayer = help.size(); moveTower(layer+1, source, help, dest); moveSlice(source, dest); moveTower(helpLayer, help, dest, source); } //-------------------------------------------------------------------------private void moveSlice(ArrayList source, ArrayList dest) { // Den Umweg ueber die Klasse Integer kann man sich sparen, aber // wir kennen Vererbung und die Klasse "Object" noch nicht. Integer slice = (Integer)source.remove(source.size()-1); dest.add(slice); ++move; print(); } //-------------------------------------------------------------------------private void print() { System.out.println("\n" + move + '.'); for (int layer=count-1; layer>=0; --layer) { System.out.print(' '); printTowerLayer(startTower, layer); System.out.print(' '); printTowerLayer(middleTower, layer); System.out.print(' '); printTowerLayer(endTower, layer); System.out.println(); } System.out.println(createString(6*count+7, '*')); } //-------------------------------------------------------------------------private void printTowerLayer(ArrayList tower, int layer) { int sliceWidth = tower.size()<=layer ? 0 : (Integer)tower.get(layer); int spaceCount = count - sliceWidth; String spaces = createString(spaceCount, ' '); String halfSlice = createString(sliceWidth, '-'); System.out.print(spaces + halfSlice + '|' + halfSlice + spaces); } //-------------------------------------------------------------------------private static String createString(int len, char c) { StringBuffer sb = new StringBuffer(len); for (int i=0; i<len; i++) { sb.append(c); } return sb.toString(); } } import java.io.BufferedReader; import java.io.InputStreamReader; public class Appl { public static void main(String[] args) { System.out.print("Anzahl Scheiben: "); try { InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); String in = reader.readLine(); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 246 / 409 int count = Integer.parseInt(in); if (count<1) { System.out.println("Fehler"); return; } Hanoi h = new Hanoi(); h.play(count); } catch (Exception x) { System.out.println("Fehler: " + x); } } } 12.10 Lsg. zu Aufgabe „Gerüchteküche“ – Kap. 12.8.2 12.10.1 Vorüberlegungen Bevor wir überhaupt mit dem Programmieren anfangen, machen wir uns Gedanken zu den Stati des Dorfes und den stabilen (End-)Zuständen. • Nicht nach jeder Simulations-Runde muss sich der Status des Dorfes ändern. Es kann z.B. sein, dass die Simulation zwei Einwohner aufeinander treffen läßt, die beide das Gerücht nicht kennen – dann kennen sie es natürlich auch hinterher noch nicht. Die AufgabenStellung verlangt, dass nur nach einer Änderung des Dorf-Status eine neue Ausgabe zu erfolgen hat – dem müssen wir Rechnung tragen. • Wann muss die Simulation beendet werden? Wenn die max. Anzahl an Durchläufen erreicht ist. Und wenn ein stabiler Endzustand erreicht ist, d.h. nichts mehr passiert. Dies ist dadurch bestimmt, dass es keinen Erzählenden mehr gibt. Achtung – es ist nicht zwingend nötig, dass alle wissend und schweigsam sind. Auch unwissende53 haben nichts zu erzählen. Also führen auch nur Unwissende und Schweigsame zu einem stabilen Endzustand. 12.10.2 Haupt-Programm Eine typische Simulation läßt sich schnell in zwei Teile zerlegen: • Erstmal muss die zu simulierende Situation aufgebaut werden, d.h. alle Daten müssen eingelesen und initiiert werden. • Dann müssen die Duchläufe gestartet werden, bis sie zu beenden sind. Damit liegt die Struktur des Haupt-Programms fest. main Anzahl Personen einlesen Anzahl max. Durchlaeufe einlesen Daten initialisieren Schleife-ueber-max-Duchlaeufe Naechste Runde der Simulation Wenn sich was geaendert hat Ausgabe Status Dorf Wenn stabiler Zustand, Schleifen-Ende 53 Gibt es die in einem echten Dorf eigentlich? © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 247 / 409 Ausgabe warum Schleifen-Ende Ein kleine Optimierung kann man noch vornehmen. todo main Anzahl Personen einlesen Anzahl max. Durchlaeufe einlesen Daten initialisieren Schleife-ueber-max-Duchlaeufe Naechste Runde der Simulation Wenn sich was geaendert hat Ausgabe Status Dorf Wenn stabiler Zustand, Schleifen-Ende Ausgabe warum Schleifen-Ende Die weiteren Erklärungen folgen (hoffentlich) später – todo... 12.10.3 Zusammenfassung Alles zusammen ergibt das damit folgendes Programm: import java.io.BufferedReader; import java.io.InputStreamReader; public class Utils { private static InputStreamReader isr = new InputStreamReader(System.in); private static BufferedReader reader = new BufferedReader(isr); public static String readString() { try { return reader.readLine(); } catch (Exception x) { } return ""; } public static int readInt() { try { String in = reader.readLine(); return Integer.parseInt(in); } catch (Exception x) { } return 0; } public static int digits(int arg) { if (arg==0) return 1; return (int)Math.log10(arg)+1; } // In java.text stehen verschiedene Format-Klassen zur Verfuegung, // mit denen diese Aufgabe auch problemlos loesbar waere. Da wir // diese Klassen aber nicht eingefuehrt haben, machen wie die // Formatierung haendisch, und lernen dabei noch etwas programmieren. public static String toString(int width, int value) { StringBuffer sb = new StringBuffer(width); int len = digits(value); for (int i=0; i<width-len; i++) { sb.append(' '); } return sb.toString() + value; } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 248 / 409 } public class Person { private static final int UNKNOWING = 1; private static final int TELLING = 2; private static final int CLOSEMOUTHED = 3; private int state = UNKNOWING; public void setTelling() { state = TELLING; } public boolean isUnknowing() { return state == UNKNOWING; } public boolean isTelling() { return state == TELLING; } public boolean isClosemouthed() { return state == CLOSEMOUTHED; } public void print() { switch (state) { case UNKNOWING: System.out.print('_'); break; case TELLING: System.out.print('|'); break; case CLOSEMOUTHED: System.out.print('*'); break; } } // // // // // Diese Funktion waere auch tabellen-gesteuert moeglich. Dann waere sie performanter. Aber so ist sie vielleicht besser zu lesen fuer einen Programmier-Anfaenger. Deshalb auch extra als Kommentar die - weil nicht notwendig - nicht zugewiesenen Stati. public static boolean meet(Person p1, Person p2) { if (p1.isUnknowing() && p2.isUnknowing()) { // p1.state = UNKNOWING; // p2.state = UNKNOWING; return false; } if (p1.isUnknowing() && p2.isTelling()) { p1.state = TELLING; // p2.state = TELLING; return true; } if (p1.isUnknowing() && p2.isClosemouthed()) { // p1.state = UNKNOWING; // p2.state = CLOSEMOUTHED; return false; } if (p1.isTelling() && p2.isUnknowing()) { // p1.state = TELLING; p2.state = TELLING; return true; } if (p1.isTelling() && p2.isTelling()) { p1.state = CLOSEMOUTHED; p2.state = CLOSEMOUTHED; return true; © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 249 / 409 } if (p1.isTelling() && p2.isClosemouthed()) { p1.state = CLOSEMOUTHED; // p2.state = CLOSEMOUTHED; return true; } if (p1.isClosemouthed() && p2.isUnknowing()) { // p1.state = CLOSEMOUTHED; // p2.state = UNKNOWING; return false; } if (p1.isClosemouthed() && p2.isTelling()) { // p1.state = CLOSEMOUTHED; p2.state = CLOSEMOUTHED; return true; } // p1.state = CLOSEMOUTHED; // p2.state = CLOSEMOUTHED; return false; } } import java.util.Random; public class Village { private Random rnd = new Random(); private int width; private Person[] persons; public Village(int count) { width = Utils.digits(count); persons = new Person[count]; for (int i=0; i<count; i++) { persons[i] = new Person(); } persons[0].setTelling(); } public boolean next() { int rnd1 = random(); int rnd2 = random(); while (rnd1 == rnd2) { rnd2 = random(); } return Person.meet(persons[rnd1], persons[rnd2]); } public boolean stableState() { for (int i=0; i<persons.length; i++) { if (persons[i].isTelling()) { return false; } } return true; } public void print() { int unknowingCount = 0; int tellingCount = 0; int closemouthedCount = 0; for (int i=0; i<persons.length; i++) { Person p = persons[i]; if (p.isUnknowing()) unknowingCount++; if (p.isTelling()) tellingCount++; if (p.isClosemouthed()) closemouthedCount++; } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 250 / 409 System.out.print( "U(" + Utils.toString(width, unknowingCount) + ") E(" + Utils.toString(width, tellingCount) + ") S(" + Utils.toString(width, closemouthedCount) + ") "); for (int i=0; i<persons.length; i++) { persons[i].print(); } } private int random() { return rnd.nextInt(persons.length); } } public class Appl { public static void main(String[] args) { System.out.println("Geruechtekueche\n===============\n"); System.out.print("Anzahl Personen: "); int count = Utils.readInt(); if (count < 2) { System.out.println("Fehler"); return; } System.out.print("Max Durchlaeufe: "); int loops = Utils.readInt(); if (loops < 1) { System.out.println("Fehler"); return; } System.out.println(); Village v = new Village(count); int width = Utils.digits(loops); printLine(width, 0, v); for (int i=0; i<loops; i++) { if (v.next()) { printLine(width, i + 1, v); if (v.stableState()) break; } } System.out.println("\nEnde wegen " + (v.stableState() ? "stabilem Zustand" : "Erreichen max Durchlaeufe")); } private static void printLine(int width, int n, Village v) { System.out.print(Utils.toString(width, n) + ": "); v.print(); System.out.println(); } } 12.11 Lsg. zu Aufgabe „Tic-Tac-Toe 1“ – Kap. 12.8.3 Bei Tic-Tac-Toe ist der Programm-Ablauf stark durch die Spiel-Regeln geprägt: es wird abwechselnd gezogen bis ein Spieler gewonnen hat oder das Spiel unentschieden zu Ende gegangen ist. Dieser Ablauf legt die Grobstruktur des Programms fest. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 251 / 409 12.11.1 Grobstruktur von „main“ Unter der Maßgabe, dass der Mensch „weiß“ spielt und damit anfängt, läßt sich der Ablauf von Tic-Tac-Toe folgendermaßen beschreiben: Endlos-Schleife über: 1. Mensch macht Zug 2. Ausgabe Spielbrett 3. Hat Mensch gewonnen? Wenn ja, Meldung und Ende 4. Ist Remis? Wenn ja, Meldung und Ende 5. Computer macht Zug 6. Ausgabe Spielbrett 7. Hat Computer gewonnen? Wenn ja, Meldung und Ende 8. --- (Remis Abfrage nicht notwendig) Da das Tic-Tac-Toe Spielfeld 9 Felder hat, kann ein Remis nur nach dem menschlichen Zug stattfinden, darum wird die Abfrage auf Remis nach dem Computer-Zug (Punkt 8) nicht benötigt. Diese Grobstruktur wirft sofort einige Fragen auf: • Das Spielbrett muss angezeigt werden. • Jemand muß wissen, wann das Spiel Remis ist. • Jemand muß wissen, wann ein Spieler das Spiel gewonnen hat. • Züge müssen berechnet, ausgeführt und verwaltet werden. Eigentlich lassen sich die Fragen leicht beantworten: • Für die Berechnung der Züge sind sicher der Mensch (im Programm vertreten durch ein einfaches Benutzer-Interface) und der Computer zuständig - beide werden in entsprechenden Klassen gekapselt. => Klasse „Human“ und „ComputerNext“ • Es ist sicher nicht Aufgabe eines Spielers, die Züge zu verwalten, Sieg und Remis zu erkennen, oder das Spielbrett anzuzeigen. Statt dessen bietet sich eine Klasse an die das Spielbrett darstellt, und über sich (Sieg, Remis, besetzte Felder, usw.) Bescheid weiß. => Klasse „Board“ Ohne an dieser Stelle der Programm-Analyse ins Detail zu gehen, lassen sich die ungefähren Verantwortlichkeiten der Klassen schon darstellen: 12.11.1.1 Klasse „Board“ Verantwortlichkeiten: • Verwaltung von Brett und der aktuellen Stellung. • Erkennen von Sieg und Remis. • Ausgabe auf der Konsole. • Annahme eines neuen Zugs. Abhängigkeiten: © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 252 / 409 • --- 12.11.1.2 Klasse „Human“ Verantwortlichkeiten: • Verwaltung seiner Spielfarbe. In der ersten Version ist dies eigentlich nicht notwendig, da der Mensch hier immer „weiß“ hat. Aber auf Dauer wird das sicher änderbar sein sollen. • Benutzer-Interaktion für die Eingabe des nächsten Zugs: 1. Zug eingeben (1..9) 2. Korrekte Eingabe? ja => weiter, nein => neue Eingabe (Sprung zu 1.) 3. Erlaubter Zug? (Feld unbesetzt?) ja => weiter, nein => neue Eingabe (Sprung zu 1.) 4. Zug zurück geben - Für die Überprüfung auf erlaubten Zug (siehe 3.) wird der Element-Funktion von „Human“ das „Board“ als Parameter übergeben. Abhängigkeiten: • Klasse „Board“, da für die Interaktion abgefragt werden muß ob der eingegebene Zug erlaubt ist oder nicht – d.h. das Feld schon besetzt ist oder nicht. 12.11.1.3 Klasse „ComputerNext“ Verantwortlichkeiten: • Verwaltung seiner Spielfarbe. In der ersten Version ist dies eigentlich nicht notwendig, da der Computer hier immer „schwarz“ hat. Aber auf Dauer wird das sicher änderbar sein sollen. • Berechung des nächsten Zugs. In der ersten Version wird hier einfach nur das nächste leere Feld zurückgegeben. Um das zu finden, wird der Element-Funktion von „ComputerNext“ das „Board“ als Parameter übergeben. Abhängigkeiten: • Klasse „Board“, da für die Berechnung die aktuelle Stellung benötigt wird, und auch abgefragt werden muß ob ein Feld schon besetzt ist oder nicht. 12.11.1.4 Klasse „Board“ II Aus den Abhängigkeiten der Klassen „Human“ und „Computer“ ergibt sich, dass die Klasse „Board“ Funktionen benötigt, um die aktuelle Stellung zu ermitteln und ob ein Zug erlaubt ist. Da wir in dieser ersten Version keine wirklich gute Computer-Spiel-Strategie implementieren wollen, und wir im Augenblick auch nicht überblicken können in welcher Form wir dafür die aktuelle Stellung benötigen, beschränken wir uns auf die Abfrage ob ein Zug erlaubt ist. 12.11.1.5 Offene Frage Nach dieser kurzen Analyse bleibt eigentlich nur noch eine Frage offen: „Wer führt den Zug © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 253 / 409 letztlich eigentlich durch?“ Lösungs-Vorschlag 1: Man könnte das Board als Parameter an die Berechnungs-Funktionen von „Human“ und „Computer“ geben, und diese geben den nächsten Zug nicht zurück, sondern setzen ihn direkt. • Direktes Setzen des Zugs geht eigentlich über die Verantwortlichkeiten der Spieler Klassen hinaus. • Mit dieser Lösung würde man jede direkte Kontrolle über den Zug abgeben, da man ihn gar nicht mehr richtig mitbekommt. Z.B. eine grafische Anzeige des Zugs oder vergleichbares wären nur schwer integrierbar. Bedenken sie, dass sowas nicht zu den Zuständigkeiten von „Human“ bzw. „ComputerNext“ gehören kann, da diese sonst in einem anderen UI nicht benutzbar wären. • Von meinem Gefühl her ist es nicht gut, dass eine Spieler-Klasse direkt am Brett rummanipuliert. Nun ist ein Stück Software zwar nicht betrügerisch (ausser es ist so programmiert), aber eine Spieler Klasse sollte nicht direkt am Brett manipulieren. Lösungs-Vorschlag 2: Man könnte dem Board die Spieler mitgeben, und das Board ruft die „nächster-Zug“ Funktion auf, und setzt den zurückgegebenen Zug dann direkt. • Dies würde Ringabhängigkeiten erzeugen: die Spieler kennen ja das Brett, und jetzt müßte das Brett auch die Spieler kennen – sehr schlecht. • Auch hier wäre die Kontrolle über den Zug – z.B. für eine grafische Rückkopplung – weg. • Und hier bekommt das Brett viel zu viel Verantwortlichkeiten – es ist für das Brett und das Umfeld zuständig, aber es soll nicht der aktive Part sein, der als zentrale Komponente das Spiel am Laufen hält (und dazu würde es bei dieser Lösung zumindest ansatzweise werden). Beide Lösungen – man legt das Setzen in die Spieler bzw. in das Brett – kommen nicht gut weg. Also lassen wir das Setzen erstmal außerhalb beider Klassen in der Spielschleife liegen. Damit hat die Haupt-Schleife auch alle Kontrolle über das Geschehen, z.B. für eine Rückkopplung auf der Kommandozeile – und die wollen wir ja auch haben. 12.11.1.6 Erste Implementierung Zuerst müssen wir die entsprechenden Objekte von Mensch, Computer und Brett in „main“ anlegen. Da die Spieler schon mal ihre Spielfarbe verwalten sollen, werden diese im Konstruktor übergeben (Integer-Konstanten aus der Klasse „Board“54). Außerdem wird einmal initial das Spielbrett ausgegeben. public class Appl { public static void main(String[] args) { Achtung – ein typischer Fall, wo ein echter Aufzählungs-Typ (siehe Kapitel 12.6) besser wäre. Da wir sie aber aus Zeitmangel nicht eingeführt haben, und sie auch erst seit JDK 1.5 verfügbar sind, lösen wir das Problem auf althergebrachte Weise mit Integer-KlassenKonstanten. 54 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 254 / 409 Board b = new Board(); Human pl1 = new Human(Board.WHITE); ComputerNext pl2 = new ComputerNext(Board.BLACK); b.print(); // Endlos-Schleife... } } Als nächstes fügen wir die Schleife von oben hinzu – schauen wir uns nochmal den Pseudocode an: Endlos-Schleife über: 1. Mensch macht Zug 2. Ausgabe Spielbrett 3. Hat Mensch gewonnen? Wenn ja, Meldung und Ende 4. Ist Remis? Wenn ja, Meldung und Ende 5. Computer macht Zug 6. Ausgabe Spielbrett 7. Hat Computer gewonnen? Wenn ja, Meldung und Ende 8. --- (Remis Abfrage nicht notwendig) 1. Mensch macht Zug Nach unserer Überlegung müssen das jetzt drei Anweisungen sein: • Objekt „Spieler“ gibt Zug zurück. • Rückkopplung im CUI über den Zug. • Setzen des Zugs am Spielbrett. Um den Zug transportieren zu können, definieren wir eine entsprechende Klasse „Move“. Achtung – wir benutzen hier nicht das Wissen, dass Spieler 1 im Augenblick immer der Mensch ist, sondern formulieren den Code schon allgemein, um für spätere Erweiterungen Erfahrungen zu sammeln und offener zu sein. Move m = pl1.next(b); System.out.println(pl1.getName() + " zieht " + m + '\n'); b.set(m); 2. Ausgabe Spielbrett Das kennen wir schon. b.print(); 3. Hat Mensch gewonnen? Oben hatten wir festgelegt, dass das Brett die Informationen über den Spielstatus hält – damit ist dieser Teil einfach hin zu schreiben. Wir übergeben aber die Spielfarbe an das Brett, damit es weiss auf wessen Sieg es checken soll. Wir wissen hier natürlich, dass wenn hier einer gewonnen hat, es nur der Mensch sein kann da er den letzten Zug gemacht hat. Aber im Sinne von allgemeiner Formulierung ist es hier sicher besser allgemein zu bleiben. Das gleiche gilt für die Ausgabe – hier fragen wir lieber allgemein den Spieler nach dem Namen statt ihn fest zu verdrahten. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 255 / 409 if (b.wins(pl1.getColor())) { System.out.println(pl1.getName() + " hat gewonnen"); return; } 4. Ist Remis? Vergleichbar zu Punkt 3. if (b.remis()) { System.out.println("Das Spiel ist remis ausgegangen"); return; } 5. Computer macht Zug 6. Ausgabe Spielbrett 7. Hat Computer gewonnen? 8. --- (Remis Abfrage nicht notwendig) Da die Punkte 1. bis 4. allgemein formuliert waren, kann der Code hierfür fast 1:1 kopiert werden. Unterschiede sind nur, dass statt Spieler 1 Spieler 2 genommen wird, und die Abfrage auf Remis entfällt. m = pl2.next(b); System.out.println(pl2.getName() + " zieht " + m + '\n'); b.set(m); b.print(); if (b.wins(pl2.getColor())) { System.out.println(pl2.getName() + " hat gewonnen"); return; } Wenn wir jetzt davon ausgehen, dass es für alle benutzten Klassen „Move, Board, ...) entsprechende Definitionen im gleichen Package gibt, und die Aufzählungs-Konstanten für die Spielfarbe in Board definiert werden, ergibt sich der ganze Quelltext: public class Appl { public static void main(String[] args) { Board b = new Board(); Human pl1 = new Human(Move.WHITE); ComputerNext pl2 = new ComputerNext(Move.BLACK); b.print(); while (true) { Move m = pl1.next(b); System.out.println(pl1.getName() + " zieht " + m + '\n'); b.set(m); b.print(); if (b.wins(pl1.getColor())) { System.out.println(pl1.getName() + " hat gewonnen"); return; } if (b.full()) { System.out.println("Das Spiel ist remis ausgegangen"); return; } m = pl2.next(b); System.out.println(pl2.getName() + " zieht " + m + '\n'); b.set(m); b.print(); if (b.wins(pl2.getColor())) { © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 256 / 409 System.out.println(pl2.getName() + " hat gewonnen"); return; } } } } Eine Bemerkung noch zum Schluss: dadurch dass wir die beiden Teile der Schleife für Mensch und Computer so ähnlich gestaltet haben, haben wir 2 Vorteile gewonnen: • Wir konnten den zweiten Teil quasi durch copy&paste erzeugen. Copy&Paste ist zwar nicht so gut wie Code nur einmal wiederverwendbar schreiben (siehe Punkt 2), aber immer noch viel besser als sich alles noch mal neu auszudenken. • Weiter nach vorne gedacht, sehen wir in welche Richtung das Zusammenführen der beiden Teile mal gehen könnte/muss. Auf Dauer kann es einfach nicht sein, dass wir zwei so ähnliche Teile Code in unserem Tic-Tac-Toe behalten, und die nicht irgendwie zusammenführen. 12.11.2 Die Rahmen der Klassen Der nächste Schritt folgt zwangsläufig ohne echtes Nachdenken. Mit unserem „main“ haben wir festgelegt welche Typen es gibt und wie ihre Schnittstelle aussieht. Das sollten wir erstmal hinschreiben55. 12.11.2.1 Klasse „Move“ Eine Kleinigkeit muss hier noch überlegt werden, bevor die Rahmen quasi von alleine da sind. Wie sieht die Klasse „Move“ aus? Nun, ein Zug besteht letztlich aus zwei Koordinaten mit den Werten 0..2 mit denen das zu setzende Feld definiert wird, und der Spielfarbe. Von daher würde eine einfache Klasse mit drei Integer-Attributen (x,y und Farbe) für die Klasse „Move“ reichen. Bevor wir die Klasse weiter durchdenken, muß ich auf einige Einwände eingehen, die sie möglicherweise haben: • Im Benutzer-Interface wird der Mensch einen Zug durch die Zahlen 1..9 eingeben. Ist ein Zug also nicht einfach nur ein Integer? Nein! Verfallen sie nie auf die Idee sich von einem Benutzer-Interface ihre interne Repräsentation der Daten vorgeben zu lassen: die Zahlen 1..9 sind unsere augenblickliche einfache Lösung zur Abbildung der Felder im Benutzer-Interface. In einem GUI benutzen sie möglicherweise 9 Buttons um das Spielfeld abzubilden, die der Benutzer zum Setzen dann direkt anklicken kann. Ist dann ein Zug etwa ein Button? • Warum die Werte 0..2 und nicht 1..3? Nun, gleiches Thema. Der Benutzer will vielleicht Koordinaten von 1..3 sehen, möglicherweise aber auch a..c – keine Ahnung, das ist Sache des Benutzer-Interfaces. Übrigens – Eclipse kann das fast von alleine für uns machen. Schauen sie sich mal die Quick-Fixes von Eclipse an. 55 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 257 / 409 Intern arbeiten wir immer – wenn möglich – 0-basiert und mit Integer-Indices, da das einfach, schnell, unproblematisch ist, und die Sprache und die Bibliotheken das auch überall so machen. • Aber warum dann nicht einfach nur einen Integer von 0..8? Das wäre doch einfacher? Nun, ein Feld auf einem Spielbrett wird nun mal durch zwei Koordinaten beschrieben. Daher steht zu erwarten, dass wir diese Repräsentation dauernd brauchen werden. Warum eine andere wählen? Und außerdem steht uns ja die Möglichkeit offen, auch noch ein zweite Schnittstelle für die andere Repräsentation anzubieten. Zurück zu unserer Klasse „Move“. Wir wissen jetzt, dass sie zwei Integer und die Farbe enthalten soll, und das wir diese sicher abfragen können müssen. Da dies trivial ist, implementieren wir die Funktionen auch direkt. public class Move { private int x; private int y; private int color; public int getX() { return x; } public int getY() { return y; } public int getColor() { return color; } } Erfüllt „Move“ damit wirklich alle Anforderungen, die aus der Main-Funktion resultieren? Nein! Wir haben eine Stelle übersehen, da wir sie noch nicht verstehen. System.out.println(pl1.getName() + " zieht " + m + '\n'); Hier wird das Move-Objekt „m“ einfach so mit Strings verkettet und ausgegeben. Nun gut, wir haben u.a. in Kapitel 9.1.2 gelernt, dass das immer geht. Aber wir wissen nicht, warum und wie. Und da wir jetzt dort eingreifen müssen – wir wollen ja unsere spezielle Tic-Tac-Toe Ausgabe des Zugs haben – müssen wir jetzt wissen, wo wir ran müssen. In Kapitel 14.11.1 werden wir lernen, dass wir hierfür die Element-Funktion „toString“ implementieren müssen. Dort lernen wir auch, wie das funktioniert, und warum das immer klappt. Hier soll uns genügen, dass wir „toString“ implementieren müssen (siehe Kapitel 12.11.4.1). Also sieht der Rahmen von „Move“ jetzt so aus, und ist jetzt wirklich fertig. public class Move { private int x; private int y; © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 258 / 409 private int color; public int getX() { return x; } public int getY() { return y; } public int getColor() { return color; } public String toString() { return null; } } 12.11.2.2 Klasse „Board“ Der erste Wurf (der Rahmen) des Klasse „Board“ fällt jetzt eigentlich einfach so ab. Die beiden Farb-Konstanten und alle augenblicklich bekannten öffentlichen Elementfunktionen sind ja durch unser Hauptprogramm vorgegeben – zusätzlich definieren wir noch eine ElementFunktion „valid“, die aufgrund der Klassen „Human“ und „Computer“ – siehe Kapitel 12.11.1.2 und 12.11.1.3 – benötigt wird. public class Board { public static final int WHITE = 1; public static final int BLACK = 2; public Board() { } public void set(Move m) { } public boolean valid(Move m) { return false; } public boolean remis() { return false; } public boolean wins(int color) { return false; } public void print() { } } Achtung – wir machen uns hier noch keine Gedanken um die Implementierung von „Board“ – das kommt später. Wir sorgen erstmal nur dafür dass unser Programm compiliert. 12.11.2.3 Klasse „Human“ Der Rahmen der Klasse „Human“ ergibt sich genauso automatisch. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 259 / 409 public class Human { public Human(int color) { } public Move next(Board b) { return null; } public int getColor() { return 0; } public String getName() { return null; } } 12.11.2.4 Klasse „ComputerNext“ Na ja, und der Rahmen der Klasse „ComputerNext“ ist fast identisch. public class ComputerNext { public ComputerNext(int color) { } public Move next(Board b) { return null; } public int getColor() { return 0; } public String getName() { return null; } } 12.11.2.5 Ergebnis Jetzt sollte unser Programm problemlos compilieren. Mehr noch nicht – gestartet macht das Verhalten des Programms wenig Sinn, da noch massenhaft Implementierungen fehlen. 12.11.3 Implementierung der Klassen Jetzt geht’s ans implementieren der Klassen. Nur, mit welcher fängt man an? Immer mit den unabhängigen Klassen, bzw. mit denen deren abhängige Klassen vorhanden sind. In unserem Fall ist das die Klasse „Move“, da diese von nichts abhängt, die Spielfeld und die Spieler-Klassen aber von ihr abhängen. Warum wählt man diese Reihenfolge? Nun, der Grund ist, dass sie so eine Klasse erhalten, die voll compilierbar, funktionstüchtig, lauffähig und testbar ist. Da sie ja von keiner anderen Klasse abhängt, bzw. diese Klassen schon implementiert sind, können sie für diese Klasse ein TestProgramm schreiben, es compilieren und laufen lassen und damit die Klasse testen. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 260 / 409 Umgekehrt geht das nicht. Würden sie in unserem Beispiel zuerst einen der Spieler entwickeln, so würde die Klasse zwar compilieren (da wir ja zufälligerweise den Rahmen von „Move“ und „Board“ schon fertig haben), aber sie könnten sie nicht testen. Damit würden sie in einem halbfertigen Zustand dann doch zu den Klassen „Move“ und „Board“ wechseln müssen, und erst nachdem diese fertig sind (vollständig fertig, mit testem und allem) könnten sie an der SpielerKlasse weiter machen. Okay – in der Praxis ist das meist nicht so schwarz/weiß. Sie benötigen für eine Klasse B nicht die vollständige Klasse A. Und viele Anforderungen an A ergeben sich auch erst, wenn die Klassen B und C A benutzen – und so wächst A parallel zu B und C. Aber jede dieser kleinen Iterationen beginnt immer bei A. 12.11.4 Klasse „Move“ Die Klasse „Move“ ist eigentlich schon fertig, da wir mit dem Rahmen (siehe Kapitel 12.11.2.1) auch direkt fast alle Element-Funktionen implementiert haben. Da sie alle einfache GetterFunktionen sind, war dies auch nicht weiter schwierig. Nur „toString“ fehlt noch. 12.11.4.1 Element-Funktion „Move.toString“ Was noch fehlt ist ein sinnvolle Darstellung eines Zugs auf der Konsole. Dazu müssen wir nur die Element-Funktion „toString“ entsprechend implementieren, z.B. so: public String toString() { return (3*x+y+1) + " - Feld " + (x+1) + '/' + (y+1); } Wie das alles funktioniert mit dieser automatischen String-Umwandlung, und warum wir „toString“ implementieren müssen, und all das – das wird in Kapitel 14.11.1 erklärt. 12.11.4.2 Konstruktoren Was der Klasse aber sicher noch fehlt, sind ein oder mehrere Konstruktoren. Ohne das wir wissen, wie Objekte der Klasse später in „Human“, „ComputerNext“ oder sonst-wo erzeugt werden, können wir schon sagen, dass es zwei mögliche Kandidaten gibt: • Move(int x, int y, int c) – x und y gehen von 0..2 Diese Variante enspricht unserer Denkweise mit x/y-Koordinate des Feldes und der Spielfarbe. Die Indices gehen von 0..2, da wir intern immer 0-basiert arbeiten. • Move(int field, int c) – field geht von 0..8 Diese Variante ist ein Zugeständnis an unser erstes einfaches Benutzerinterface. Hier muss der Wert 1..9 in die Felder 0..2/0..2 umgesetzt werden. Da dies möglicherweise öfter vorkommt, implementieren wir diese Zerlegung zentral in der Klasse „Move“. Da intern aber immer 0-basiert gerechnet wird, erwartet die Funktion einen Feld-Index von 0..8. Da es nicht viel Arbeit ist, implementieren wir profilaktisch beide Konstruktoren. Man könnte mit © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 261 / 409 dieser Arbeit auch problemlos warten, bis sie denn wirklich gebraucht werden. So, im Vorfeld implementiert, kann es halt passieren, dass die Schnittstelle doch nicht paßt, oder wir gar eine Funktion implementieren, die wir dann doch nicht brauchen. public Move(int n, int c) { x = n/3; y = n%3; color = c; } public Move(int x, int y, int c) { this.x = x; this.y = y; color = c; } 12.11.4.3 Zusammenfassung Damit ergibt sich der komplette Quelltext der Klasse „Move“. Source „Move.java“ public class Move { private int x; private int y; private int color; public Move(int n, int c) { x = n/3; y = n%3; color = c; } public Move(int x, int y, int c) { this.x = x; this.y = y; color = c; } public int getX() { return x; } public int getY() { return y; } public int getColor() { return color; } public String toString() { return (3*x+y+1) + " - Feld " + (x+1) + '/' + (y+1); } } 12.11.5 Klasse „Board“ Nach „Move“ ist die nächst-tiefere Klasse „Board“, d.h. wird die als nächstes implementiert. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 262 / 409 12.11.5.1 Spielfeld-Repräsentation und erweiterte Klassen-Definition von „Board“ Die erste Entscheidung, die zu treffen ist, ist: wie repräsentieren wir das Spielfeld in der Klasse? Schauen wir uns mal die Anforderungen an das Spielfeld an: • Wahlfreier Zugriff wird benötigt: zwei Indices definieren das Feld • Sortierung wird nicht benötigt. • Spielfeld benötigt keine dynamische Größe. • Suchen findet nicht statt. • Inhalte sollten nicht umsortiert werden – das Brett ist und bleibt wie es ist. Gleichen wir dies mit unserem Java Wissen ab, dann sehen wir das ein normales 2-dim Array die Anforderungen am Besten erfüllt. Stellt sich noch die Frage, was wird der Inhalt des Arrays sein, d.h. wie repräsentieren wir im Programm ein Feld bzw. den Status eines Feldes? Welche Zustände kann ein Feld annehmen? • Leer • Weißer Spielstein • Schwarzer Spielstein Das klingt nach einer Aufzählung mit Klassen-Konstanten56. Und Konstanten für die Spielfarben sind schon vorhanden – ergänzen wir noch eine für ein leeres Feld. Damit sind die Attribute und Klassen-Konstanten von „Board“ klar. public class Board { public static final int EMPTY = 1; public static final int WHITE = 2; public static final int BLACK = 3; private int[][] field; ... } 12.11.5.2 Attribut-Initialisierung und Konstruktor „Board“ Bei der Objekt-Initialisierung muss nur das Spielfeld aufgebaut werden, d.h. ein 3x3 Array, bei dem alle Felder „Empty“ sind. Es gibt viele Arten dies zu machen, hier mache ich das mit einer einfachen Attribut-Initialisierung. Der Konstruktor bleibt dadurch leer. public class Board { private static final int EMPTY = 1; private static final int WHITE = 2; private static final int BLACK = 3; private int[][] field = { EMPTY, EMPTY, EMPTY { EMPTY, EMPTY, EMPTY { EMPTY, EMPTY, EMPTY { }, }, } Achtung – noch ein Fall, wo ein echter Aufzählungs-Typ (siehe Kapitel 12.6) besser wäre. Da wir sie aber aus Zeitmangel nicht eingeführt haben, und sie auch erst seit JDK 1.5 verfügbar sind, lösen wir das Problem auf althergebrachte Weise mit Integer-Klassen-Konstanten. 56 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 263 / 409 }; public Board() { } ... } 12.11.5.3 Element-Funktion „Board.set“ Die Elementfunktion „set“ bekommt einen Zug übergeben und setzt ihn. Hierbei gibt es ein Thema, über das man zumindest kurz nachdenken sollte: Was passiert, wenn als Zug ein Feld übergeben wird, das schon besetzt ist? Wir definieren hier für den ZugParameter, dass ein solcher Parameter nicht erlaubt ist. An dieser Stelle ist das eine akzeptable Forderung, da im Rahmen des „main“-Designs die von den Spielern „berechneten“ Züge regelkonform sein müssen. Trotzdem sollten wir uns darüber im klaren sein, dass ein Programmier-Fehler in den SpielerKlassen hier zu einem Problem führen kann. Solchen Situationen kann man z.B. mit Assertions begegnen, die wir aber aus Zeitmangel im Rahmen des Tutorials nicht besprechen. public void set(Move m) { field[m.getX()][m.getY()] = m.getColor(); } 12.11.5.4 Element-Funktion „Board.valid“ Die Elementfunktion „valid“ soll zurückgeben, ob ein Spielzug erlaubt ist, d.h. das Feld noch unbesetzt ist. public boolean valid(Move m) { return field[m.getX()][m.getY()]==EMPTY; } 12.11.5.5 Element-Funktion „Board.remis“ Die Element-Funktion „remis“ soll zurückgeben, ob das Spiel remis ausgegangen ist. Die erste Idee das zu lösen, ist ganz einfach: es müssen alle Felder besetzt sein. boolean remis() { for (int x=0; x<3; ++x) { for (int y=0; y<3; ++y) { if (field[x][y]==EMPTY) return false; } } return true; } Aber dann denken wir noch mal darüber nach. Diese Funktion gibt auch dann „true“ zurück, wenn das Spielfeld voll ist, aber ein Spieler gewonnen hat. Hier wird nämlich nicht auf Remis untersucht, sondern auf „Erzwungendes-Spielende wegen vollem Feld“. Also eigentlich müßten wir vorher abfragen, ob kein Spieler gewonnen hat – erst dann arbeitet die Funktion korrekt. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 264 / 409 boolean remis() { if (wins(WHITE) || wins(BLACK)) { return false; } for (int x=0; x<3; ++x) { for (int y=0; y<3; ++y) { if (field[x][y]==EMPTY) return false; } } return true; } Wenn wir uns jetzt aber unser Haupt-Programm vor Augen führen, dann sehen wir, dass das nicht das ist, was wir wollten. Uns ging es schon um das erzwungende Spielende, wenn das Brett voll ist, denn einen möglichen Sieg hatten wir vorher schon abgehandelt. if (b.wins(pl1.getColor())) { System.out.println(pl1.getName() + " hat gewonnen"); return; } if (b.remis()) { System.out.println("Das Spiel ist remis ausgegangen"); return; } Jetzt haben wir zwei Möglichkeiten: 1. Wir akzeptieren die obige aufwendigere „remis“ Funktion, selbst wenn dort Abfragen gemacht werden, die in unserem Programm nicht notwendig sind. 2. Oder wir bleiben bei der ersten Variante, benennen die Funktion aber um. Achtung – kommen Sie bitte nicht auf die Idee, den Namen bei „remis“ zu belassen, aber von der Funktionalität her „Brett-voll“ zu implementieren – die Funktion macht dann nicht mehr das, was der Name verspricht. Irgendwann wird sich das rächen. Man muss nur im HauptProgramm die Abfrage-Reihenfolge von „wins“ und „remis“ verändern, und schon hat man den Salat. Aus Sicht des Haupt-Programms wäre dann noch alles okay – in der Praxis leider nicht. Ich persönlich habe mich für Möglichkeit 2 entschieden, d.h. ich nehme unsere erste einfachere Variante der Element-Funktion, und benenne sie in „full“ um. Denn das liefert sie zurück: ob das Brett voll ist. public boolean full() { for (int x=0; x<3; ++x) { for (int y=0; y<3; ++y) { if (field[x][y]==EMPTY) return false; } } return true; } Hinweis – wenn ich jetzt doch noch eine Element-Funktion „remis“ benötige, dann würde ich sie jetzt auf Basis von „wins“ und „full“ ganz einfach implementieren können. So habe ich zwei Fliegen mit einer Klappe geschlagen: zur Zeit keine überflüssigen Implementierungen, und auf Dauer einfache verständliche Funktionen. So z.B. könnte „remis“ dann aussehen: boolean remis() { if (wins(WHITE) || wins(BLACK)) { return false; © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 265 / 409 } return full(); } 12.11.5.6 Element-Funktion „Board.wins“ Die Element-Funktion „wins“ soll zurückgeben, ob der Spieler der übergebenen Farbe gewonnen hat. Möglicherweise gibt es einen tollen Algorithmus um das schnell und kurz zu checken – wenn ja, so kenne ich ihn nicht. Und ich hatte auch keine Lust mir groß Gedanken darum zu machen – acht mögliche Gewinnstellungen kann ich auch schnell bruce-force durchtesten. public boolean wins(int if (field[0][0]==c && if (field[1][0]==c && if (field[2][0]==c && if (field[0][0]==c && if (field[0][1]==c && if (field[0][2]==c && if (field[0][0]==c && if (field[0][2]==c && return false; } c) { field[0][1]==c field[1][1]==c field[2][1]==c field[1][0]==c field[1][1]==c field[1][2]==c field[1][1]==c field[1][1]==c && && && && && && && && field[0][2]==c) field[1][2]==c) field[2][2]==c) field[2][0]==c) field[2][1]==c) field[2][2]==c) field[2][2]==c) field[2][0]==c) return return return return return return return return true; true; true; true; true; true; true; true; 12.11.5.7 Element-Funktion „Board.print“ Die Element-Funktion „print“ soll das Spielfeld ausgeben. Kein Problem – zwei geschachtelte for-Schleifen für das 2-dim Array und darin eine switch-Anweisung für die jeweiligen FeldAusgaben. public void print() { for (int x=0; x<3; ++x) { for (int y=0; y<3; ++y) { switch (field[x][y]) { case EMPTY: System.out.print('_'); break; case WHITE: System.out.print('O'); break; case BLACK: System.out.print('X'); break; } } System.out.println(); } System.out.println(); } 12.11.5.8 Zusammenfassung Damit ergibt sich der komplette Quelltext der Klasse „Board“. Source „Board.java“ public class Board { public static final int EMPTY = 1; public static final int WHITE = 2; © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 266 / 409 public static final int BLACK = 3; private int[][] field = { EMPTY, EMPTY, EMPTY { EMPTY, EMPTY, EMPTY { EMPTY, EMPTY, EMPTY }; { }, }, } public Board() { } public void set(Move m) { field[m.getX()][m.getY()] = m.getColor(); } public boolean valid(Move m) { return field[m.getX()][m.getY()]==EMPTY; } public boolean full() { for (int x=0; x<3; ++x) { for (int y=0; y<3; ++y) { if (field[x][y]==EMPTY) return false; } } return true; } public boolean wins(int if (field[0][0]==c && if (field[1][0]==c && if (field[2][0]==c && if (field[0][0]==c && if (field[0][1]==c && if (field[0][2]==c && if (field[0][0]==c && if (field[0][2]==c && return false; } c) { field[0][1]==c field[1][1]==c field[2][1]==c field[1][0]==c field[1][1]==c field[1][2]==c field[1][1]==c field[1][1]==c && && && && && && && && field[0][2]==c) field[1][2]==c) field[2][2]==c) field[2][0]==c) field[2][1]==c) field[2][2]==c) field[2][2]==c) field[2][0]==c) return return return return return return return return true; true; true; true; true; true; true; true; public void print() { for (int x=0; x<3; ++x) { for (int y=0; y<3; ++y) { switch (field[x][y]) { case EMPTY: System.out.print('_'); break; case WHITE: System.out.print('O'); break; case BLACK: System.out.print('X'); break; } } System.out.println(); } System.out.println(); } } 12.11.6 Klasse „Human“ 12.11.6.1 Erweiterte Klassen-Definition von „Human“ Bzgl. der Attribute der Klasse „Human“ fällt erstmal nur auf, dass sie sich die Spielfarbe merken © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 267 / 409 muss. public class Human { private int color; ... } 12.11.6.2 Attribut-Initialisierung und Konstruktor „Human“ Der Konstruktor muss einfach nur das Attribut mit der übergebene Spiel-Farbe initialisieren. public class Human { private int color; public Human(int c) { color = c; } ... } 12.11.6.3 Element-Funktion „Human.getColor“ Die Element-Funktion „getColor“ soll nur die Spiel-Farbe zurückgeben – trivial. public int getColor() { return color; } 12.11.6.4 Element-Funktion „Human.getName“ Die Element-Funktion „getName“ soll den Namen des Spielers zurückgeben – auch trivial. public String getName() { return "Mensch"; } 12.11.6.5 Element-Funktion „Human.next“ Die einzige Funktion der Klasse „Human“ in der wirklich was passiert, ist die Element-Funktion „next“. Sie ist auch relativ kompliziert – was aber nicht an der eigentlichen Funktionalität liegt, sondern daran dass in ihr Benutzer-Interaktion statt findet – und „Fehlertolerantes Einlesen ist einfach der pure Spass!“. Bevor wir anfangen zu implementieren lassen sie uns kurz überlegen, was in „next“ passieren muss: 1. Ausgabe „Bitte Zug eingeben“ 2. Benutzer-Eingabe 1..9 3. Korrekte Eingabe? Wenn nein, Fehlerhinweis und zurück zu 1. 4. Eingabe in Zug wandeln 5. Freies Feld? Wenn nein, Fehlerhinweis und zurück zu 1. 6. Rückgabe des Zugs © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 268 / 409 Man sieht, dass man hier eine Endlos-Schleife benötigt, die im Falle einer richtigen Eingabe abgebrochen wird. import java.io.BufferedReader; import java.io.InputStreamReader; public class Human { ... public Move next(Board b) { InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); while (true) { System.out.print("Bitte machen Sie ihren Zug (1..9): "); try { String in = reader.readLine(); int field = Integer.parseInt(in); if (field<1 || field>9) { System.out.println("Fehlerhafte Eingabe!\n"); continue; } Move m = new Move(field-1, color); if (b.valid(m)) { System.out.println(); return m; } System.out.println("Feld schon besetzt!\n"); } catch (Exception x) { System.out.println("Fehlerhafte Eingabe!\n"); } } } } 12.11.6.6 Zusammenfassung Damit ergibt sich der komplette Quelltext der Klasse „Human“. Source „Human.java“ import java.io.BufferedReader; import java.io.InputStreamReader; public class Human { private int color; public Human(int c) { color = c; } public int getColor() { return color; } public String getName() { return "Mensch"; } public Move next(Board b) { InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 269 / 409 while (true) { System.out.print("Bitte machen Sie ihren Zug (1..9): "); try { String in = reader.readLine(); int field = Integer.parseInt(in); if (field<1 || field>9) { System.out.println("Fehlerhafte Eingabe!\n"); continue; } Move m = new Move(field-1, color); if (b.valid(m)) { System.out.println(); return m; } System.out.println("Feld schon besetzt!\n"); } catch (Exception x) { System.out.println("Fehlerhafte Eingabe!\n"); } } } } 12.11.7 Zwischenstand Hiermit könnten sie jetzt schon das Tic-Tac-Toe eingeschränkt zum Laufen bringen. Sie ersetzen im Haupt-Programm einfach den Computer-Spieler durch einen zweiten menschlichen Spieler, und schon ist der Computer das Spielfeld für zwei Menschen beim Tic-Tac-Toe. public class Appl { public static void main(String[] args) { Board b = new Board(); Human pl1 = new Human(Move.WHITE); Human pl2 = new Human(Move.BLACK); // <<<<<<<<<<<<<< Aenderung b.print(); ... } } 12.11.8 Klasse „ComputerNext“ Die erste Implementierung eines Computer-Spielers wird ganz einfach ausfallen – der Computer wird einfach das erste leere Feld nehmen. Das Attribut „color“ und die Element-Funktionen „ComputerNext.getColor“ und „ComputerNext.getName“ werden analog zur Klasse „Human“ sein – d.h. können wir hier auf die Herleitung verzichten. 12.11.8.1 Element-Funktion „ComputerNext.next“ Damit bleibt als einzige Funktion mit etwas Aufwand die Element-Funktion „next“ über. Aber auch diese ist nicht schwer. Der Computer soll einfach das erste freie Feld nehmen, dass heisst wir müssen einfach nur die © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 270 / 409 Felder durchprobieren, und den ersten erlaubten Zug zurückgeben. Da wir in der Klasse „Move“ u.a. einen Konstruktor implementiert haben, der mit einem Feld-Index auskommt, können wir uns hier auf eine Schleife beschränken, statt zwei verschachtelte Schleifen implementieren zu müssen. public Move next(Board b) { for (int i=0; true; i++) { Move m = new Move(i, color); if (b.valid(m)) { return m; } } // <-- keine Rueckgabe noetig, da man hier nicht hinkommen kann } Auch hier verlassen wir uns wieder darauf, dass das Spiel in der Gesamtheit sauber und fehlerfrei programmiert ist. Wir ignorieren das Problem: welchen Zug geben wir zurück, wenn es kein freies Feld mehr gibt – da dieses in der Praxis nicht vorkommen dürfte. Da – unter dieser Bedingung – spätestens beim Feld-Index „8“ ein freies Feld gefunden werden muß, können wir uns die Abbruch-Bedingung sparen. Damit wird die Stelle hinter den Schleifen nie erreicht werden können. Das erkennt der Compiler, und verlangt daher dort von uns auch keine Rückgabe – wir würden eh nur sinnloses Zeug dort hinschreiben können. 12.11.8.2 Zusammenfassung Damit ergibt sich der komplette Quelltext der Klasse „ComputerNext“. Source „ComputerNext.java“ public class ComputerNext { private int color; public ComputerNext(int c) { color = c; } public int getColor() { return color; } public String getName() { return "Computer (naechstes freies Feld)"; } public Move next(Board b) { for (int i=0; true; i++) { Move m = new Move(i, color); if (b.valid(m)) { return m; } } } } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 271 / 409 12.11.9 Fazit Das Programm ist fertig. Natürlich gibt es noch eine Menge offener Wünsche – diese ergeben sich zum größten Teil aber aus der Aufgaben-Stellung: das Programm hat kein GUI, der Computer spielt ziemlich dumm spielt, die Gegner liegen fest, der Mensch beginnt immer, usw. Wirklich unschön an dem Programm sind eigentlich nur wenige Dinge: • Die Schleife im Haupt-Programm enthält zweimal fast den gleichen Code. • Es gibt keine saubere Trennung von Benutzer-Oberfläche und Programm-Logik. • Das Programm enthält viele implizite Annahmen – siehe z.B. Kapitel 12.11.5.3. In den nächsten Kapiteln werden wir die ersten beiden Unschönheiten beseitigen und das TicTac-Toe gleichzeitig um einige Features bereichern. Um angemessen mit impliziten Annahmen und darauf basierenden Fehlern umgehen zu können, müßten wir uns mit Assertions auseinander setzen – dies wird leider im Rahmen der Vorlesung aus Zeitmangel nicht gemacht. Wenn sie sich das gesamte Programm mal anschauen, so sollten sie eins bemerken. Jede Klasse ist genau für eine Sache zuständig, und die meisten Funktionen sind ziemlich einfach und erledigen genau eine Aufgabe. Kompliziert und aufwändig sind nur die Funktionen, die Benutzer I/O machen – vor allem Benutzer-Eingabe. Dies ist nicht ungewöhnlich. • Durch ein sauberes OO-Design (OOD) erhält man klare überschaubare Abstraktionen, und die Klassen und Funktionen bleiben einfach, klein und übersichtlich. • Benutzer I/O dagegen, und hier vor allem die Eingabe ist nicht trivial. Denn Benutzer können allen möglichen Blödsinn eingeben, d.h. die Eingabe muss fehlertolerant sein und gute Meldungen liefern. Dies ist nicht imnmer trivial. 13 Packages Packages sind neben Klassen das zweite Modul-Konzept von Java. Während Klassen wiederverwendbare Komponenten darstellen und einen Aspekt der realen Welt abbilden, sind Packages ein reines Strukturierungsmittel, das konzeptionell zusammengehörige Klassen zu einer Einheit verbindet. 13.1 Package-Anweisung In Java ist jede Klasse automatisch Bestandteil eines Packages - indem sie eine .java Datei mit einer Package-Anweisung wie z.B. „package packagename;“ beginnen, geben sie automatisch das Package an, zu dem die Klasse gehört. package mypackage; // Package Anweisung public class A { } // Die Klasse A liegt jetzt im Package „mypackage“ Achtung – eine Package-Anweisung muss immer die erste Anweisung in einem Quelltext sein. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 272 / 409 Daher vorher darf es nur Kommentar- und Leerzeilen geben. Hinweis – Klassen in Quelltexten ohne Package-Anweisungen liegen im sogenannten DefaultPackage – siehe Kapitel 13.5. Hinweis – denken sie daran, dass jedes Package auf der Datei-Systeme-Ebene einem Verzeichnis entsprechen muss – siehe auch Kapitel 3.8, Kapitel 4.2 bzw. Kapitel 4.3.2. Hinweis – per Konvention sollten Package Namen immer klein beginnen und klein geschrieben werden, z. B.: „nameeinespackage“ – siehe auch Kapitel 3.2.5. 13.2 Klassen in Packages Der vollständige Name einer Klasse ist nicht nur der Klassen-Name, sondern beinhaltet auch den bzw. die Package-Namen, durch den Punkt-Operator getrennt – Beispiel „java.util.Date“. Dies wird auch der „vollständig-referenzierte“ oder auch der „vollständig-qualifizierte“ Name genannt. Um eine Klasse zu benutzen gibt es drei Möglichkeiten: • Benutzung des vollständig qualifizierten Namens. • Benutzung einer Import-Anweisung für die Klasse. • Benutzung einer Import-Anweisung für das gesamte Package der Klasse. 13.2.1 Benutzung des vollständig qualifizierten Namens Sie können immer jede Klasse über ihren voll referenzierten Namen ansprechen. java.util.Date d = new java.util.Date(); 13.2.2 Benutzung einer Import-Anweisung Sie können eine Klasse in den Namensraum ihrer Datei importieren. Dafür können sie mit dem Schlüsselwort „import“ Import-Anweisungen an den Anfang ihrer Datei schreiben. ImportAnweisungen müssen nach der Package-Anweisung stehen, wenn eine solche vorhanden ist. Aber sie müssen vor jeder Klassen- oder Interface-Definition erfolgen. Mit import und exakt referenzierter Klasse importieren sie eine Klasse: import java.util.Date; ... Date d = new Date(); Alternativ können sie mit einer Import-Anweisung auch alle Symbole eines Package in den Namensraum ihrer Datei importieren. Hierfür muss statt des Klassen-Namens in der ImportAnweisung ein „*“ angegeben werden. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 273 / 409 import java.util.*; ... Date d = new Date(); Vector v = new Vector(); 13.2.3 Klassen im gleichen Package Klassen im gleichen Package brauchen weder importiert noch vollständig qualifiziert werden – sie sind immer automatisch bekannt und können einfach durch Benutzung des KlassenNamens angeprochen werden – siehe z.B. Bsp. todo. 13.3 Language-Package Z. B. die Klassen String und Object sind Teil des Packages java.lang, trotzdem konnten wir sie benutzen57 ohne dieses Package importiert zu haben. Das Package java.lang wird immer automatisch importiert, ohne dass Sie sich darum kümmern müssen. 13.4 Verschachtelung Packages können natürlich wieder in einander verschachtelt sein. Wollen Sie z. B. eine Klasse Model dem Package generator im Package report zuordnen, so müssen sie nur folgende package-Anweisung am Anfang Ihrer Datei Model.java unterbringen: package report.generator; Wollen sie diese Klasse in einem anderen Package nutzen, so haben sie natürlich folgende drei Möglichkeiten: report.generator.Model m = new report.generator.Model(); import report.generator.Model; ... Model m = new Model(); import report.generator.*; ... Model m = new Model(); 13.5 Default-Package Und was passiert, wenn sie keine Package-Anweisung benutzen? Nichts - auch das ist korrekter Code. 57 Falls Ihnen nicht klar ist, das Sie Object benutzt haben - denken Sie daran, dass Object automatisch die Basisklasse einer Klasse ist, wenn keine andere explizit angegeben ist. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 274 / 409 Für kleinere Projekte bzw. schnelle Tests wurde zur Arbeitserleichterung in Java das Feature eingeführt, dass in diesem Fall alle diese Klasse in einem Default-Package der Entwicklungsumgebung landen. Sie brauchen auch keinen import-Anweisung anzugeben - das Default-Package wird immer automatisch importiert. Dem Java-Compiler ist die physikalische Realisation des Default-Packages übrigens freigestellt - er braucht nur eins, kann aber auch mehrere Packages anlegen und diese dann über mehrere Unterverzeichnisse zu verteilen. 13.6 Verzeichnisse Packages müssen physikalisch auf der Platte durch Unterverzeichnisse abgebildet werden. Eine Klasse „first.second.third.ClassName“ muss also in einer Datei „ClassName.java“ in den Verzeichnis-Struktur „first/second/third“ liegen. Moderne Entwicklungsumgebungen wie z.B. Eclipse können die notwendige VerzeichnisStruktur automatisch im Hintergrund erzeugen, und legen z.B. neue Dateien automatisch richtig ab. 14 Vererbung 14.1 Vererbungs-Hierarchien Vererbung bedeutet "ist ein". Abb. 14-1 : Vererbungs-Hierarchie Voraussetzung für öffentliche Vererbung ist immer eine ist-ein Beziehung. Allgemein: Abb. 14-2 : Allgemeine Vererbungs-Hierarchie Hierbei ist: • A Basisklasse (von B und C) • B ist abgeleitet von A => B ist ein A => alles was für A gilt, gilt auch für B • C ist abgeleitet von A => C ist ein A => alles was für A gilt, gilt auch für C In Richtung der abgeleiteten Klassen findet eine Spezialisierung statt: • B ist eine Spezialisierung von A • Ein Pferd ist eine Spezialisierung eines Säugetiers • Alles, was für Säugetiere gilt (z. B. geboren werden, schwanger sein, sterben), gilt auch für Pferde. All diese Attribute und Funktionen erbt Pferd von Säugetier. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 275 / 409 In Richtung der Basis-Klassen findet eine Generalisierung oder Verallgemeinerung statt - Basis-Klassen fassen gemeinsame Dinge der abgeleiteten Klassen zusammen: • A enthält alle Gemeinsamkeiten von B und C • Vogel enthält alles Vogel-typische, unabhängig, ob es sich um eine Amsel oder eine Möve handelt. Hinweise • Abgeleitete Klassen (z. B. B und C) sind unabhängig voneinander. • Von einer Klasse können beliebig viele andere Klassen abgeleitet werden. • Eine Klasse hat immer nur eine Basisklasse (Einfachvererbung58). Achtung Unterscheiden sie zwischen ”hat ein” bzw. ”ist implementiert mit” und ”ist ein”. • Eine Person hat einen Namen, ist aber kein Name => Aggregation, Komposition, ... • Eine Person ist ein Lebewesen, immer ohne wenn und aber => Vererbung 14.2 Implementation Wie wird Vererbung in Java implementiert? Syntax [Modifier] class klassenname extends Basisklasse { Klassen-Definition } public class A { public void afct() { System.out.println("afct in A"); } } public class B extends A { public void bfct() { System.out.println("bfct in B"); } } // Normales A Verhalten A a = new A(); a.afct(); // Normales B Verhalten B b = new B(); b.bfct(); // Aber B ist auch ein A, darum 'kann' es alles, was A 'kann' b.afct(); // Ausgabe: afct in A 58 Es gibt viele objektorientierte Sprachen, die auch Mehrfachvererbung unterstützen. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 276 / 409 Abb. 14-3 : Vererbungs-Hierarchie des Beispiels An diesem Beispiel sehen sie, dass die Klasse B alle Funktionen von A erbt, d.h. sie ohne weitere Schreibarbeit zur Verfügung stehen. Jede Änderung in A wirkt sich damit sofort auch auf B (und natürlich alle weiteren abgeleiteten Klassen) aus. Und sie sehen, dass die Klasse B das Verhalten (Methoden) von A erbt, d.h. sie ohne weitere Schreibarbeit zur Verfügung stehen. Jede Änderung in A wirkt sich damit sofort auch auf B (und natürlich alle weiteren abgeleiteten Klassen) aus. 14.3 Schlüsselwort super In jeder Element-Funktion steht das Schlüsselwort super zur Verfügung, das immer für den Objektanteil der Basisklasse des aktuellen Objektes steht59. Beispiel siehe z. B. Konstruktoren. 14.4 Konstruktoren Konstruktoren werden in Java nicht vererbt. Dies macht auch keinen Sinn, da ein Konstruktor immer das erzeugte Objekt initialisieren soll - ein geerbter Konstruktor aber nur den Objektanteil der Basisklasse initialisieren kann. Sie müssen daher für jede Klasse wieder neu Konstruktoren erstellen. Die Konstruktoren der abgeleiteten und der Basis-Klasse sind automatisch miteinander verkettet, d.h. jeder Konstruktor einer abgeleiteten Klasse ruft als erstes defaultmäßig den Standard-Konstruktor der Basisklasse auf. public class A { public A() { System.out.println("Konstruktor A"); } } public class B extends A { public B() { System.out.println("Konstruktor B"); } } B b = new B(); Ausgabe Konstruktor A Konstruktor B Sie sehen, dass als erstes automatisch der Standard-Konstruktor der Basisklasse aufgerufen wird. Wollen sie, dass ein anderer Konstruktor für die Basisklasse benutzt wird, können sie 59 Ähnlich wie this, das immer für das komplette aktuelle Objekt steht (siehe Kapitel 12.4.3). © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 277 / 409 diesen am Anfang des Konstruktors der abgeleiteten Klasse mit super angeben. public class A { public A(int i) { System.out.println("Konstruktor A mit " + i); } } public class B extends A { public B() { super(11); System.out.println("Konstruktor B"); } } Ausgabe Konstruktor A mit 11 Konstruktor B Wenn die Basisklasse keinen Standard-Konstruktor hat, so müssen sie in den abgeleiteten Konstruktoren mit super einen Konstruktor angeben. Sie können in einem Konstruktor mit this auch einen anderen Konstruktor der Klasse anspringen, der dann einen Konstruktor der Basisklasse aufruft. Achtung – die Verwendung von überschriebenen (s.u.) Element-Funktionen in Konstruktoren ist gefährlich, da die Oberklassen-Anteile noch nicht konstruiert sind. 14.5 finalize Finalize Funktionen (siehe Kapitel 12.3) werden in Java nicht automatisch verkettet. Wenn sie dies erreichen wollen, so müssen sie in der finalize Funktion die finalize Funktion der Basisklasse selber aufrufen. public class B extends A { protected void finalize() { super.finalize(); } } 14.6 Überschreiben Funktionen von Klassen können in abgeleiteten Klassen überschrieben werden, d.h. sie können für eine abgeleitete Klasse neu definiert werden (gleicher Name, gleiche Parameterliste). public class A { public void fct() { System.out.println("fct in A"); } } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 278 / 409 public class B extends A { public void fct() { System.out.println("fct in B"); } } A a = new A(); a.fct(); B b = new B(); b.fct(); // fct in A // fct in B Auf die Art und Weise kann eine nicht passende Implementierung einer Basisklasse in einer abgeleiteten Klasse neu implementiert, d. h. überschrieben werden. Ein Beispiel wäre die Methode berechneGehalt in einer Klasse Angestellter und in der abgeleiteten Klasse Vertreter. Ein Vertreter ist sicherlich auch ein Angestellter, d.h. für ihn gelten die Methode getName, getPersonalNo(),..., aber das Gehalt wird bei Vertretern oft anders berechnet. Oft ist es so, dass die Basisklassen Implementierung gar nicht so schlecht ist, aber eben nicht 100 % passt. Vielleicht bekommt der Vertreter zusätzlich zu einem Festgehalt nur noch einen variablen Anteil hinzu – in diesem Fall wäre die Basisklassen Implementierung ja nicht falsch, sondern eben nur ein Teil der korrekten Implementierung. Darum ist es oft sinnvoll in einer Neu-Implementierung auf die Basisklassen Implementierung zurückzugreifen. Hierbei gibt es zwei ‚reine‘ Formen (Korrektur und Filterung), aber natürlich auch beliebige Mischformen. 14.6.1 Korrektur Falls das Ergebnis der Basisklassen-Elementfunktion nicht 100% passend ist, kann es in einer abgeleiteten Klasse korrigiert werden – denken sie an das Vertreter Beispiel von oben. Prinzip void f() { super.f(); // Korrektur } // expliziter Aufruf der Original-Elementfunktion // Korrektur des Ergebnisses 14.6.2 Filterung Falls die Basisklassen-Elementfunktion nicht alle Fälle (korrekt) behandelt, können diese vorher abgefangen und behandelt werden. Prinzip void f() { // Filterung super.f(); } // Filterung mancher Faelle // expliziter Aufruf der Original-Elementfunktion Die Filterung bezieht sich häufig auf die Übergabeparameter. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 279 / 409 14.7 Ist-ein Beziehung Eine Konsequenz aus der Semantik ‘Vererbung ist eine ist-ein Beziehung’ ist, dass einer Referenz-Variablen der Basisklasse auch ein Objekt einer abgeleiteten Klasse zugewiesen werden kann. // Klasse B ist von A abgeleitet A a1 = new A(); A a2 = new B(); Dies ist ganz im Sinne der Semantik. Ist-ein heisst, dass für ein B Objekt alles gilt, was für ein A Objekt gilt - und daher ein B Objekt auch das Interface von A unterstützt. Bemerkung - falls sie das Ganze etwas verwundert, machen sie sich mal von der ganzen Computerei frei, und betrachten das Ganze mit einem normalen Beispiel: Wenn sie z.B. auf einen Stuhl zeigen und sagen „das ist ein Stuhl“, dann wird ihnen wohl niemand widersprechen. Aber auch die Aussage „das ist ein Möbelstück“ wäre ohne Frage richtig. Und genau das gleiche passiert hier: Die Referenz-Variable „a2“ sagt mit ihrem statischen Typ „A“ das sie ein Möbelstück referenziert (darauf zeigt), obwohl sie doch in Wirklichkeit einen Stuhl (ein Objekt vom Typ „B“) referenziert (darauf zeigt). Aber daran ist nichts falsches und unwahres - sie sagt nur nicht alles. Aber in vielen Kontexten reicht das. Wir sagen zu unserem Besuch auch „Nimm dir einen Stuhl“, und lassen offen ob er sich in einen Sessel, die gute Coach oder den normalen Holzstuhl setzen soll. Warum auch? Im Prinzip würde es sogar reichen zu sagen „Nimm doch bitte Platz“. Hinweis - man unterscheidet daher auch in den sogenannten statischen und den dynamischen Typ. • Der statische Typ ist der Typ der Referenz-Variablen. Diesen Typ sieht der Compiler, da er ohne wenn und aber zur Compilezeit feststeht und eindeutig bekannt ist - er steht ja als Typ an der Definition der Referenz-Variablen. Im Beispiel ist dies der Typ “A“ der ReferenzVariablen „a1“ und „a2“. • Dem gegenüber ist der dynamische Typ der echte Typ des Objekts, auf das verwiesen wird. Dieser ist zur Compilezeit nicht zwingend bekannt, und muß nicht dem statischen Typ entsprechen. Im Beispiel ist der dynamische Typ des von „a1“ referenzierten Objekts „A“, während es bei „a2“ „B“ ist. 14.8 Polymorphie Überschreiben und ist-ein-Beziehung zusammen ermöglichen ein Feature, das das Schlüsselkonzept aller OO Sprachen ist: Polymorphie. public class A { public void fct() { System.out.println("fct in A"); } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 280 / 409 } public class B extends A { public void fct() { System.out.println("fct in B"); } } A a1 = new A(); A a2 = new B(); a1.fct(); // fct in A a2.fct(); // fct in B - obwohl ueber eine A Referenz-Variab. aufgerufen In Java wird der Funktionsaufruf erst zur Laufzeit festgelegt60, und zwar in Abhängigkeit vom echten Typ des Objekts, und nicht in Abhängigkeit vom Typ der Referenz-Variablen. Dieses Sprachfeature wird mit Polymorphie bezeichnet. Mit Polymorphie ist gemeint, dass eine Funktion vielgestaltig ist, d. h. in Abhängigkeit vom Kontext unterschiedlich (angepasst) reagiert. Genau genommen reagiert natürlich nicht eine Funktion unterschiedlich, sondern es werden unterschiedliche Funktionen aufgerufen, ohne dass sich der Entwickler um die echten Objekt-Typen und deren verschiedene FunktionenImplementierungen kümmern muss. Dies ermöglicht es ihm, ähnliche Objekte61 gleich zu behandeln, ohne Details kennen zu müssen (z.B. welche Klassen es gibt, wie sie heissen, wie sie zu behandeln sind, usw...). Hinweis – im ersten Augenblick sieht Polymorphie nicht nach was besonderem aus, sondern eher nur nach einem kleinen Sprachgag – aber dies ist falsch. Es ist das Schlüsselkonzept der Objektorientierung. Seine wahre Mächtigkeit erkennt man meist erst in praktischen Einsätzen, von denen in den weiteren Kapiteln viele folgen werden. Ein kleines Beispiel als Einstimmung folgt gleich – siehe Kapitel 14.12. 14.9 abstract Es gibt Situation, in denen eine Basisklasse keine sinnvolle Default-Implementierung für eine Element-Funktion anbieten kann – ein Beispiel hierfür findet sich u.a. im Beispiel-Kapitel 14.12. In diesem Fall bekommt die entsprechende Element-Funktion den Modifier abstract, was bedeutet, dass diese Element-Funktion in dieser Klasse nur deklariert, aber nicht implementiert wird. Sobald mindestens eine Element-Funktion in einer Klasse abstract ist (und sei es auch durch Vererbung – siehe Klasse „B“ im Beispiel), muss auch die Klasse den Modifier abstract bekommen. In einer tieferen abgleiteten Klasse ohne Modifier abstract muss diese ElementDieses Verhalten wird u.a. auch als dynamische Bindung, späte Bindung oder late binding bezeichnet. 61 Ähnliche Objekte sind Objekte, die eine gemeinsame Basisklasse (oder in Java auch ein gemeinsames Interface – siehe todo...) haben. 60 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 281 / 409 Funktion jetzt überschrieben und implementiert werden. public abstract class A { public abstract void f(); } public abstract class B extends A { } public abstract class C extends B { public abstract void f(); } public class D extends C { public void f() { System.out.println("Hallo"); } } A a = new D(); a.f(); // Ausgabe: Hallo Eine abstrakte Klasse kann damit automatisch nicht mehr instanziiert werden – was ja auch keinen Sinn mehr macht, da sie eine Funktion ohne Implementierung anbietet. public abstract class A { public abstract void f(); } A a = new A(); a.f(); // Fehler – A laesst sich nicht instanziieren, da abstract // Welche Funktion sollte das dann auch sein?? Eine Klasse darf auch dann abstrakt sein, wenn sie keine abstrakten Funktionen hat. Typischerweise tritt dieser Fall bei Klassen auf, die keine konkreten Objekte beschreiben, sondern nur allgemeine Beschreibungen für die Gemeinsamkeiten einer „Objekt-Familie“ sind. 14.10 Casts und instanceof Mit Casts kann man statische Typen in der Vererbungs-Hierarchie verschieben. Dazu wird der gewünschte Zieltyp in Klammern vor den Quellausdruck geschrieben. Achtung – dies geht nur, solange die referenzierten Objekte wirklich solche sind. Ansonsten wird eine Exception geworfen. Mit dem Operator „instanceof“ kann abgefragt werden, ob ein Objekt von einem bestimmten Typ ist. public class A { } public class B extends A { © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 282 / 409 public void f() { System.out.println("B.f"); } } A var = new A(); if (var instanceof B) { System.out.println("Variable var referenziert ein B Objekt"); B b = (B)var; b.f(); } else { System.out.println("Variable var referenziert KEIN B Objekt"); } var = new B(); if (var instanceof B) { System.out.println("Variable var referenziert ein B Objekt"); B b = (B)var; b.f(); } else { System.out.println("Variable var referenziert KEIN B Objekt"); } Ausgabe Variable var referenziert KEIN B Objekt Variable var referenziert ein B Objekt B.f try { A a = new A(); B b = (B)a; b.f(); } catch (Exception x) { System.out.println(x); } Ausgabe java.lang.ClassCastException Achtung – Casts innerhalb einer Vererbungs-Hierarchie sind in einer Sprache mit häufig untypisierten Schnittstellen62 relativ normal (vergleiche hier z.B. die untypisierten Container, die wir in Kapitel 9.3 kennen gelernt haben63). Trotzdem sollte ihnen klar sein, dass Casts kein guter Programmierstil sind, und auf das Notwendigste beschränkt sein sollten. Noch extremer ist dies mit der Verwendung von „instanceof“. Im Normallfall sollte die Kenntnis der konkreten Typen hinter einem Basis-Klasse oder einem Interface unnötig sein, da dies z.B. die Erweiterbarkeit und Wiederverwendbarkeit stark einschränkt. Daher sollte die Benutzung von „instanceof“ der gut begründete Ausnahmefall bleiben. Also ganz untypisiert sind sie nicht, sondern typisiert auf „Object“ – siehe Kapitel 14.11. Aber abgesehen von den elementaren Datentypen sind alle Klassen von „Object“ abgeleitet, so dass die Typisierung hier nicht wirklich viel hilft. 63 Mit dem JDK 1.5 hat Java Generics bekommen, die u.a. typisierte Container ermöglichen. Damit werden Casts in Java etwas weniger häufig notwendig, was sicher zu einer Verbesserung der Progamm-Qualität beitragen wird. Aus Zeitgründen werden wir in der Vorlesung Generics nicht besprechen. 62 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 283 / 409 14.11 Klasse „java.lang.Object“ Alle Klassen in Java sind immer direkt oder indirekt von der Klasse „Object“ abgeleitet. Wenn sie keine Basisklasse angeben, wird automatisch „Object“ als Basisklasse angesetzt. Die Klasse „Object“ beinhaltet allgemeine Methoden, die für jede Klasse sinnvoll sind, z.B. clone() equals(Object) finalize() getClass() hashCode() toString() Erzeugt eine flache Kopie des Objekts. Dazu muss das Objekt das Interface Cloneable implementieren und diese Funktion überschreiben. Arrays sind immer kopierbar. Vergleicht zwei Objekte auf Identität, d.h. Referenzgleichheit. Für Objektvergleiche, d.h. tiefe Vergleiche bzw. ein spezielles Vergleichsverhalten muss diese Funktion überschrieben werden. Die normale finalize Methode Gibt die ‘Meta-Klasse’ zum Objekt zurück. Gibt einen Hash-Wert für das Objekt zurück. Gibt das Objekt in einer Text-Repräsentation zurück. Diese Funktion wird automatisch z.B. bei Ausgaben auf die Console oder bei Wandlungen in einen String aufgerufen. Siehe auch Kapitel 14.11.1. Wenn sie die Funktionen für ihre Klassen anpassen wollen bzw. müssen, d.h. die geerbte Funktionalität nicht ausreicht, müssen sie sie überschreiben. Hinweis – die Funktionen „equals“ und „hashCode“ sind nicht ganz unabhängig voneinander. wenn sie die Equals-Funktion überschreiben, müssen sie auch die HashCode-Funktion entsprechend überschreiben. Näheres hierzu finden sie z.B. in der offiziellen Java-Doku oder vielen Büchern. Aus Zeitmangel wird dies in der Vorlesung nicht besprochen. Achtung – das automatische Erben von „Object“ gilt nicht für Interfaces (Kapitel 14.13), sondern nur für Klassen. Da „Object“ eine Klasse ist, können Interfaces nicht von ihr erben, d.h. Klassen haben immer genau eine absolute Basisklasse - das ist „Object“. Interfaces sind da anders. 14.11.1 Element-Funktion „Object.toString“ Ich möchte hier noch mal besonders auf die Funktion „toString“ hinweisen. Sie wird immer dann automatisch aufgerufen, wenn eine Wandlung von einem Objekt in einen String notwendig ist. Dies geschieht: • Bei der Ausgabe eines Objekts auf der Console mit „System.out.print“ bzw. „System.out.println“. • Bei der Verkettung eines Strings mit einem Objekt mit dem Plus-Operator. Da die Klasse „java.lang.Object“ eine Default-Implementierung anbietet, kann jedes Objekt © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 284 / 409 immer ausgegeben bzw. in einen String gewandelt werden. Achtung – die DefaultImplementierung von Object liefert keinen besonders sinnvolle Text-Repräsentation, wie auch? public class A { } public class Appl { public static void main(String[] args) { A a = new A(); System.out.println(a); String s = "String: " + a; System.out.println(s); } } mögliche Ausgabe A@119c082 String: A@119c082 Mit dem Überschreiben der Funktion „toString“ legen sie das Ergebnis der Wandlung fest. public class A { public String toString() { return "Ich bin ein A-Objekt"; } } public class Appl { public static void main(String[] args) { A a = new A(); System.out.println(a); String s = "String: " + a; System.out.println(s); } } Ausgabe Ich bin ein A-Objekt String: Ich bin ein A-Objekt 14.11.2 Object als Basistyp Jede Klasse ist in Java direkt oder indirekt von „java.lang.Object“ abgeleitet. Daher kann jedes Objekt in Java immer einer Referenz-Variablen vom statischen Typ „Object“ zugewiesen werden. Beispiele: Object o1 = new java.util.TreeMap(); Object o2 = new StringBuffer(); Object o3 = new javax.swing.JFrame(); Von daher ist „Object“ der kleinste gemeinsame Nenner aller Objekte – wir lassen die elementaren Datentypen mal außen vor. Und von daher werden in Java viele Funktionen auf „Object“ typisiert, zum Beispiel die Funktionen der Container-Klassen aus Kapitel 9.3. Daher werden in der Praxis häufig Up-Casts in der Klassen-Hierarchie (siehe Kapitel 14.10) benötigt – siehe auch die Beispiele der Container-Klassen in Kapitel 9.3. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 285 / 409 14.12 Anwendung – Beispiel „Obstkorb“ 14.12.1 Aufgabe Nehmen wir an, sie wollen einen Obstkorb implementieren: • Ein Obstkorb soll einfach mehrere Früchte verschiedener Obstsorten aufnehmen können. • Jede Frucht hat einen Namen. • Auch der Obstkorb hat einen Namen. • Außerdem soll der Obstkorb einen Konsolen Ausgabe folgender Form haben: • Name vom Obstkorb • Anzahl der Früchte im Obstkorb • Darstellung alle Früchte – alphabetisch sortiert nach dem Namen der Frucht • Die Darstellung einer Frucht besteht aus Name und Obstsorte. • Für den Anfang begnügen wir uns mit den zwei Obstsorten „Apfel“ und „Birne“. Hier eine mögliche Beispiel-Ausgabe eines Obstkorbs mit 5 Früchten: Gewünschte Ausgabe – wenn denn der Obstkorb fertig wäre... Ich bin der Obstkorb "Geschenk" und enthalte 5 Fruechte: - Bauchiger Adler (Birne) - Dickes Schwein (Apfel) - Fetter Kohl (Birne) - Gruener Baum (Apfel) - Saftiger Schmatz (Apfel) Vorgehen – um zu sehen, wie uns Vererbung, „ist-ein“-Beziehung und Polymorphie hier helfen, werden wir das Programm erstmal ohne diese Sprachmittel implementieren, und dann Stück für Stück Sprachmittel für Sprachmittel nutzen, und dann hoffentlich sehen wie sie uns das Programmierer-Leben erleichtern. Bemerkung – wem ein Obstkorb mit Früchten zu abstrakt oder zu gesund ist, der möge sich statt dessen eine Angestellten-Verwaltungs-Software für Arbeiter und Vertriebler vorstellen, oder eine Flughafen-Dispostions-Verwaltung für Flugsteige und Tankwagen, oder oder oder... 14.12.2 Lösung 1 – ohne Vererbung und ohne Polymorphie Zuerst brauchen wir Klassen für Äpfel und Birnen, die den Namen halten und sich selber entsprechend der Aufgabenstellung darstellen können. public class Apple { private String name; public Apple(String n) { name = n; } public void print() { System.out.println("- " + name + " (Apfel)"); } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 286 / 409 public String getName() { return name; } } public class Pear { private String name; public Pear(String n) { name = n; } public void print() { System.out.println("- " + name + " (Birne)"); } public String getName() { return name; } } Bevor wir zum Obstkorb kommen – dem eigentlichen Knackpunkt des Programms – implementieren wir die „main“ Funktion – und bekommen damit implizit die SchnittstellenDefinition vom Obstkorb. public class Appl { public static void main(String[] args) { FruitBasket fb = new FruitBasket("Geschenk"); fb.insert(new Apple("Dickes Schwein")); fb.insert(new Pear("Fetter Kohl")); fb.insert(new Apple("Saftiger Schmatz")); fb.insert(new Apple("Gruener Baum")); fb.insert(new Pear("Bauchiger Adler")); fb.print(); } } Dann brauchen wir den Obstkorb selber, und das wird schwieriger. Da aber die Schnittstelle aus dem „main“ automatisch heraus fällt, fangen wir damit an: public class FruitBasket { private String name; public FruitBasket(String n) { name = n; } public void insert(Apple apple) { ... } public void insert(Pear pear) { ... } public void print() { ... } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 287 / 409 } Jetzt stellt sich die Frage, wie wir Äpfel und Birnen im Obstkorb speichern können – am besten schon alphabetisch sortiert. Für die dynamische Speicherung von Objekten haben wir in Kapitel todo mehrere Container kennengelernt. U.a. gab es dabei auch eine Klasse „TreeMap“, die über einen Schlüssel (hier bei uns der Name) alphabetisch sortiert. Hinweis – in der Praxis würde man hierfür natürlich niemals einen assoziativen Container (d.h. einen mit Schlüssel/Wert Paaren) nehmen, sondern einen Container, der die Sortierung automatisch auf den Objekten selber vornimmt. Den gibt es in Java natürlich auch, z.B. mit der Klasse „TreeSet“ in „java.util“, aber a) kennen wir ihn nicht, und b) müßten wir für seinen Gebrauch wissen, wie man unsere Äpfel und Birnen sortierbar bekommt, d.h. wir müßten mit dem Interfaces „Comparable“ arbeiten oder einen eigenen Comparator implementieren. Für beides ist es noch etwas früh – von daher nehmen wir hier die schlechtere Lösung mit der Klasse „TreeMap“. Da wir aber noch ohne Vererbung und „ist-ein“ Beziehung arbeiten, können wir Äpfel und Birnen nicht in einer TreeMap speichern64. Also geben wir der Obstkorb-Klasse für jeden Obsttyp eine eigene TreeMap. import java.util.TreeMap; public class FruitBasket { private String name; private TreeMap apples = new TreeMap(); private TreeMap pears = new TreeMap(); public FruitBasket(String n) { name = n; } public void insert(Apple apple) { apples.put(apple.getName(), apple); } public void insert(Pear pear) { pears.put(pear.getName(), pear); } public void print() { int count = apples.size(); count += pears.size(); System.out.println("Ich bin der Obstkorb \"" + name + "\" und enthalte " + count + " Fruechte:"); ... } } Als Problem bleibt jetzt nur noch die Ausgabe der Früchte – damit sie über alle Füchte Okay, in Java geht es schon, da beide implizit von java.lang.Object abgeleitet sind, und TreeMap mit Objects arbeitet. Aber letztlich wäre das dann auch nur die Verwendung von Vererbung und „ist-ein“ Beziehung – und das wollen wir ja noch nicht. 64 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 288 / 409 alphabetisch ist, müssen beide TreeMaps parallel durchlaufen werden und die jeweils kleinere Frucht (bezogen auf den Namen) muß ausgegeben werden. Das klingt kompliziert – gerade wenn man an später mit noch mehr Obstsorten und noch mehr TreeMaps denkt – daher sehe ich hier von einer Lösung ab und überlasse diese dem Studenten ;-) 14.12.3 Lösung 2 – immer noch ohne Vererbung und ohne Polymorphie Wenn Lösung 1 bei der Ausgabe zu kompliziert ist, man aber keine Vererbung und Polymorphie zur Verfügung hat – was macht man dann? Nun, wenn man in Java ein Problem hat, dann macht man eine Klasse daraus. Das Problem ist hier, dass wir Äpfel und Birnen gleichzeitig verwalten wollen, dass aber noch nicht können bzw. hier mehr wollen. Also schreiben wir eine Klasse, die das für uns macht – und da sie Früchte verwaltet, nennen wir sie „Fruit“. public class Fruit { private Apple apple = null; private Pear pear = null; public Fruit(Apple a) { apple = a; } public Fruit(Pear p) { pear = p; } public String getName() { if (apple!=null) { return apple.getName(); } return pear.getName(); } public void print() { if (apple!=null) { apple.print(); return; } pear.print(); } } Die Klassen „Apple“, „Pear“ und „Appl“ sind von dieser Änderung nicht betroffen, nur der Obstkorb selber, da er jetzt intern diese Hilfsklasse „Fruit“ benutzt. import java.util.TreeMap; import java.util.Iterator; public class FruitBasket { private String name; private TreeMap fruits = new TreeMap(); public FruitBasket(String n) { name = n; } public void insert(Apple apple) { © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 289 / 409 fruits.put(apple.getName(), new Fruit(apple)); } public void insert(Pear pear) { fruits.put(pear.getName(), new Fruit(pear)); } public void print() { int count = fruits.size(); System.out.println("Ich bin der Obstkorb \"" + name + "\" und enthalte " + count + " Fruechte:"); Iterator i = fruits.values().iterator(); while (i.hasNext()) { Fruit f = (Fruit)i.next(); f.print(); } } } Bevor sie weitergehen zu Lösung 3, schauen sie sich Lösung 2 ruhig mal in Ruhe an. Wir haben hier einen wichtigen Grundsatz von Java kennen gelernt, und sehen ein großes Problem bisheriger Programmierung. Der Grundsatz ist einfach: „Haben sie ein Problem, machen sie eine Klasse draus“. Im Prinzip ist dies der alte Grundsatz der Software-Entwicklung „Es gibt kein Problem, das sich nicht durch eine weitere Indirektion kleiner machen läßt“ in neuen Gewändern, nämlich dem OO-Kleid. Ein großes Problem unseres kleinen Programms ist die Wartbarkeit: immer wenn wir hier eine neue Obstsorte einführen, müssen wir neben einer weiteren Klasse in unserem kleinen Programm schon mehrere Code-Stellen mit ändern: • Die Klasse „FruitBasket“ braucht eine weitere „insert“ Funktion. • Und die Klasse „Fruit“ muss quasi überall angepaßt werden. Und wir haben ja nur ein kleines Programm. Und dass es so relativ wenig lokalisierte Stellen sind, liegt eigentlich schon daran, dass wir die fast gesamte Logik für die Obstsorten in der Klasse „Fruit“ gesammelt haben. In einem wirklich großen ernsthaften Programm würde bei einer klassischen Programmierung – wie wir sie hier haben – das Ändern (Einfügen, Löschen, Modifizieren) z.B. einer FlughafenRessource oft an tausenden von Stellen Code-Änderungen nach sich ziehen. Ein horrender Aufwand, bei dem man sich nie wirklich ganz sicher sein kann, alle relevanten Stellen berücksichtigt zu haben. 14.12.4 Lösung 3 – mit Vererbung, aber noch ohne Polymorphie Im Prinzip hat sich eine bessere Lösung schon die ganze Zeit aufgedrängt, aber wir haben sie bewußt nicht eingesetzt. Mit Vererbung und den Konsequenzen über „ist-ein“ Beziehung sind einige Dinge viel einfacher – wir können jetzt an vielen Stellen Äpfel und Birnen gleich behandeln. Die Klasse „Fruit“ bekommt hier einen ganz anderen Zweck: statt fast alle Probleme zu kapseln © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 290 / 409 mutiert sie zu einer sehr einfachen Basisklasse, die – da sie nicht instanziierbar sein soll – abstrakt ist: public abstract class Fruit { } Am Beispiel des Apfels schauen wir uns den kleinen Unterschied in der Klassen-Definition der Obstsorten an – er besteht nur aus der Angabe der Basisklasse und dem Schlüsselwort „extends“. public class Apple extends Fruit { private String name; public Apple(String n) { name = n; } public void print() { System.out.println("- " + name + " (Apfel)"); } public String getName() { return name; } } Und wie sieht der Obstkorb jetzt aus? import java.util.TreeMap; import java.util.Iterator; public class FruitBasket { private String name; private TreeMap fruits = new TreeMap(); public FruitBasket(String n) { name = n; } public void insert(Apple apple) { fruits.put(apple.getName(), apple); } public void insert(Pear pear) { fruits.put(pear.getName(), pear); } public void print() { int count = fruits.size(); System.out.println("Ich bin der Obstkorb \"" + name + "\" und enthalte " + count + " Fruechte:"); Iterator i = fruits.values().iterator(); while (i.hasNext()) { Fruit f = (Fruit) i.next(); if (f instanceof Apple) { Apple a = (Apple) f; a.print(); continue; } if (f instanceof Pear) { Pear p = (Pear) f; p.print(); continue; } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 291 / 409 } } } Unsere TreeMap kann jetzt direkt alle Früchte ohne Wrapper-Klasse aufnehmen – siehe Element-Funktionen „insert“. Und die Fall-Unterscheidung zwischen Äpfeln und Birnen findet sich nur noch in der Schleife der Obstkorb-Ausgabe. 14.12.5 Lösung 4 – mit Vererbung, aber immer noch ohne Polymorphie Als wir Vererbung eingeführt haben, haben wir noch nicht über Polymorphie gesprochen, sondern statt dessen den Vorteil herausgestellt, dass gemeinsame Funktionen in eine Basisklasse gelegt werden, und abgeleitete Klassen diese einfach erben. Diesen Vorteil können wir hier auch nutzen, in dem wir das Namens-Handling von Äpfeln und Birnen in die Basisklasse „Fruit“ verschieben – hier am Beispiel von „Apple“ verdeutlicht. public abstract class Fruit { private String name; public Fruit(String n) { name = n; } public String getName() { return name; } } public class Apple extends Fruit { public Apple(String n) { super(n); } public void print() { System.out.println("- " + getName() + " (Apfel)"); } } Dies hat den Nebeneffekt, dass sich im Obstkorb die beiden „insert“ Element-Funktionen zu einer zusammenfassen lassen: public void insert(Fruit fruit) { fruits.put(fruit.getName(), fruit); } 14.12.6 Lösung 5 – mit Vererbung und mit Polymorphie Als unschöne Stelle im Programm bleibt die Ausgabe-Funktion des Obstkorbs über. public void print() { int count = fruits.size(); System.out.println("Ich bin der Obstkorb \"" + name + "\" und enthalte " + count + " Fruechte:"); Iterator i = fruits.values().iterator(); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 292 / 409 while (i.hasNext()) { Fruit f = (Fruit) i.next(); if (f instanceof Apple) { Apple a = (Apple) f; a.print(); continue; } if (f instanceof Pear) { Pear p = (Pear) f; p.print(); continue; } } } Sie sieht kompliziert und fehlerträchtig aus, und sie ist wartungsintensiv. Das Problem hier ist, dass wir Objekte unterschiedlicher Klassen in die Hand bekommen und wir jeweils die entsprechende Element-Funktion der jeweiligen Klasse benutzen wollen. Aber genau das liefert uns doch die Polymorphie! Wir müssen in der Klasse „Fruit“ nur die entsprechende „print“ Funktion zur Verfügung stellen, und sie dann in den abgeleiteten Klassen überschreiben. Und da wir für „print“ in „Fruit“ keine sinnvolle Default-Implementierung kennen, machen wir die Funktion „abstract“. public abstract class Fruit { private String name; public Fruit(String n) { name = n; } public String getName() { return name; } public abstract void print(); } Für die Obst-Klassen „Apple“ und „Pear“ ändert sich gar nichts. Aber die Ausgabe-Funktion des Obstkorbs ist auf einmal ganz einfach: public void print() { int count = fruits.size(); System.out.println("Ich bin der Obstkorb \"" + name + "\" und enthalte " + count + " Fruechte:"); Iterator i = fruits.values().iterator(); while (i.hasNext()) { Fruit f = (Fruit) i.next(); f.print(); } } Und sie enthält keine Abhängigkeiten auf die verwendeten Obstsorten mehr – auf einmal ist sie ganz allgemeingültig. 14.12.7 Erweiterte Aufgabe Lassen sie sich das mal auf der Zunge zergehen. In unserem ganzen Programm gibt es fast © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 293 / 409 keine Abhängigkeiten mehr auf die verwendeten Obstsorten. Was heißt das für uns, wenn wir z.B. eine neue Sorte „Bananen“ einführen wollen? Zuerst brauchen wir in Anlehnung an „Apple“ und „Pear“ eine weitere Klasse „Banana“. Die neue Klassen ist vollkommen unabhängig vom restlichen Programm, und wir können sie einfach hinzufügen: public class Banana extends Fruit { public Banana(String n) { super(n); } public void print() { System.out.println("- " + getName() + " (Banane)"); } } Und in „main“ benutzen wir sie einfach: public static void main(String[] args) { FruitBasket fb = new FruitBasket("Geschenk"); fb.insert(new Apple("Dickes Schwein")); fb.insert(new Pear("Fetter Kohl")); fb.insert(new Apple("Saftiger Schmatz")); fb.insert(new Apple("Gruener Baum")); fb.insert(new Pear("Bauchiger Adler")); fb.insert(new Banana("Krumme Wurst")); fb.insert(new Banana("Langer Lulatsch")); fb.print(); } Ausgabe Ich bin der Obstkorb "Geschenk" und enthalte 7 Fruechte: - Bauchiger Adler (Birne) - Dickes Schwein (Apfel) - Fetter Kohl (Birne) - Gruener Baum (Apfel) - Krumme Wurst (Banane) - Langer Lulatsch (Banane) - Saftiger Schmatz (Apfel) Und ansonsten müssen wir nichts machen – das Programm funktioniert einfach! Unser eigentliches Programm – hier die Klasse „Obstkorb“ – arbeitet nur auf der Basisklasse, und muß – dank Polymorphie – die konkreten Klassen nicht kennen, und ist daher vollkommen unabhängig. Das Programm läßt sich so sehr leicht verändern oder erweitern. 14.13 Interfaces Interfaces sind spezielle Java-Klassen: • Sie sind quasi eine auf die Spitze getriebene abstrakte Klasse. • Sie werden mit dem Schlüsselwort interface deklariert. • Sie können nur abstrakte Funktionen und Klassen-Variablen enthalten, d.h. sie enthalten weder Implementationen noch Attribute. Häufig enthalten sie nur KlassenKonstanten – siehe Kapitel 12.5.2. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 • • • • • • Seite 294 / 409 Sie haben keine implizite Basisklasse. Sie müssen keine Funktionen enthalten, d.h. sie dürfen auch leer sein. Von Interface’s werden Klassen mit dem Schlüsselwort implements abgeleitet. Eine Klasse kann beliebig viele Interface’s implementieren. Es können Referenz-Variablen vom Typ des Interfaces definiert werden. Interfaces benötigen – als quasi spezielle Klassen – ihre eigene Quelltext-Datei, die natürlich so heißen muß wie das Interface. public interface Inter { public void fct(); } public class Imple implements Inter { public void fct() { System.out.println("In fct() von Impl"); } } Inter in = new Imple(); in.fct(); Interface’s sind quasi das Java-Sprachmittel für Mehrfachvererbung, wobei eine Klasse nur über die extends Schiene eine Implementierung erben kann. Hinweis – in Java ist es möglich, dass ein Interface komplett leer ist. Mit dem Implementieren eines solchen Interfaces bekommt eine Klasse quasi ein Flag, dass irgendein Verhalten gewünscht ist, okay ist, nicht sein soll, oder was auch immer. Ein Beispiel hierfür ist das Interface „Serializable“, mit dem eine Klasse serialisierbar wird – siehe Kapitel todo. 14.14 Modul-Entkopplung Vererbung und Polymorphie sind auch ein Mittel um Module voneinander zu entkoppeln, d.h. die Module unabhängig voneinander zu machen. Schauen wir uns hierzu mal ein Beispiel an: Das Benutzer-Interface unterliegt in der Praxis oft häufigeren Änderungen. Damit Änderungen im Benutzer-Interface nicht Änderungen im gesamten Programm nach sich ziehen, versucht man die eigentliche Programm-Logik und das Benutzer-Interface in Schichten aufzuteilen65. Abb. 14-4 : 2 Schicht-Architektur eines Programms In einer solchen Architektur gibt es eine klare Abhängigkeits-Beziehung: hier kennt das UI66 die In der Praxis findet man häufig drei Schichten oder mehr. Bei einer 3 Schicht-Architektur werden die Schichten für das Benutzer-Interface und die Programm-Logik meist noch um eine Schicht zur Daten-Haltung ergänzt. 66 UI – User-Interface, d.h. Benutzer-Interface 65 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 295 / 409 BL67, aber umgekehrt nicht! Die BL stellt Dienste zur Verfügung, die vom UI benutzt werden können. Die BL kennt aber kein konkretes UI, und darf daher auch keine Benutzer-Interaktion ausführen! Abb. 14-5 : Abhängigkeit in einer 2 Schicht-Architektur Was ist aber nun, wenn z.B. die BL während der Arbeit Eingaben benötigt, z.B. einen Dateinamen, einen Parameter, ein Passwort oder sonstwas? Sie darf ja keine BenutzerInteraktion ausführen, und kann das UI auch nicht dazu auffordern (da sie es nicht kennt)68. Was macht man dann? Formulieren wir das Problem etwas anders, vielleicht fällt dann die Lösung leichter: Die BLSchicht darf nicht selber Benutzer-Interaktion machen, d.h. muss sie dies indirekt machen, z.B. mittels eines Funktions-Aufrufs. Da sie die-UI Schicht nicht kennt, kann sie dort keine Funktion direkt aufrufen. Sie muss also eine Funktion aufrufen, die in der BL-Schicht bekannt ist, aber in einer unbekannten UI-Schicht ausgeführt wird. Sehen sie die Lösung? • In der BL-Schicht muss eine Funktion bekannt sein, damit die BL-Schicht sie nutzen kann, aber sie kann in der BL-Schicht nicht implementiert sein. • In der UI-Schicht muss genau diese Funktion dann implementiert sein, und sie muss quasi über die BL-Funktion aufrufbar sein. Das klingt doch wie Vererbung und Polymorphie: • In der BL-Schicht wird ein Interface benötigt, dass die Parameter-Hol Funktion definiert. • In der UI-Schicht muss sich eine UI-Klasse von diesem Interface ableiten und die Parameter-Hol Funktion implementieren. • Z.B. beim Aufruf der BL-Schicht könnte jetzt das Objekt, das das BL-Interface implementiert, mitgegeben werden. Abb. 14-6 : Design der Modul-Entkopplung Und hier das Ganze als Quelltext: package bl; public interface GetUserParameterInterface { public String get(); } package bl; public class BusinessLogic { BL – Business-Logic, d.h. Geschäfts- oder Programm-Logik In vielen Programm findet man hier dann häufig Code, der die Schichtung mit ihrer sauberen Abhängigkeit durchbricht, und alle Ansätze zur Trennung und Modularisierung ad-acta legt. Dies kann natürlich nicht Sinn der Sache sein. 67 68 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 296 / 409 public String doit(GetUserParameterInterface gupi) { String parameter = gupi.get(); return "\"" + parameter + "\""; } } package ui; import java.io.BufferedReader; import java.io.InputStreamReader; import bl.BusinessLogic; import bl.GetUserParameterInterface; public class UserInteraction implements GetUserParameterInterface { private BusinessLogic bl = new BusinessLogic(); public void doit() { System.out.println("Starte Verarbeitung..."); String s = bl.doit(this); System.out.println("Ergebnis: " + s); } public String get() { try { System.out.println("Bitte geben sie einen String ein:"); InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); return reader.readLine(); } catch (Exception x) { } return ""; } } import ui.UserInteraction; public class Appl { public static void main(String[] args) { UserInteraction ui = new UserInteraction(); ui.doit(); } } mögliche Ausgabe Starte Verarbeitung... Bitte geben sie einen String ein: Hallo Welt Ergebnis: "Hallo Welt" Hinweis – um die Trennung zwischen BL- und UI-Schicht sauber darzustellen, sind die entsprechenden Klassen (bzw. Interfaces) in entsprechende Packages „bl“ und „ui“ eingefügt. Bemerkung – ein weiteres, vielleicht etwas einsichtigeres Beispiel findet sich in der Praktikums-Aufgabe 14.16.2. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 297 / 409 14.15 Fazit Wenn sie Vererbung und Polymorphie einsetzen, dann hat das Hinzufügen oder Entfernen oder Verändern einer Klasse fast keine Auswirkungen auf ihr eigentliches Programm – ausgenommen sind die Stellen an denen die Objekte erzeugt werden, aber auch da kann man sich helfen. Immer wenn sie es schaffen ihr Programm, ihre Programm-Struktur oder ihre ProgrammArchitektur auf Baisklassen (oder Interfaces) aufzubauen, dann haben sie gewonnen. Sie haben leichtes Spiel mit Veränderungen, ohne dass sie ihren ganzen Code nach Änderungen durchforsten müssen. Vererbung und Polymorphie sind die Schlüsselkonzepte von OO! 14.16 Aufgaben 14.16.1 Aufgabe „Obstkorb“ Implementieren sie das Obstkorb Programm aus Kapitel 14.12.6. Erweitern sie es um eine Klasse für Zitronen (Englisch „lemon“). Lösung siehe Kapitel 14.17. 14.16.2 Aufgabe „ProgressBar“ Ein typisches Problem der Art von Kapitel 14.14 („Modul-Entkopplung“) ist eine FortschrittsAnzeige („ProgressBar“). In der BL Schicht findet eine Verarbeitung statt, die längere Zeit braucht. Damit der Benutzer eine Rückkopplung bekommt, soll eine Fortschritts-Anzeige aufgeblendet werden. Nur, die BL-Schicht kennt keine UI-Schicht und weiss auch nicht, was für eine Fortschritts-Anzeige im jeweiligen UI-Kontext sinnvoll ist. Daher bietet sich eine Entkopplung über ein Interface an. • Denken sie sich eine einfache Verarbeitung in der BL-Schicht aus, die etwas Zeit kostet. Z.B. eine einfache Schleife mit einem „Thread.Sleep“ (siehe Kapitel 11.3.1). • Schreiben sie ein erstes kleines Programm ohne Fortschritts-Anzeige mit BL- und UISchicht. • Definieren sie ein Interface, mit dem eine Fortschritts-Anzeige betrieben werden kann. • Implementieren sie eine Fortschritts-Anzeige. • Integrieren sie Interface und Fortschritts-Anzeige in ihr Programm. • Implementieren sie eine weitere Fortschritts-Anzeige. Zeigen sie durch einen einfachen Austausch der Fortschritts-Anzeigen dass die BL-Schicht mit beiden betrieben werden kann, ohne dass die BL-Schicht betroffen ist (d.h. geändert werden muss). • Mögliche Fortschritts-Anzeigen auf Konsolen-Ebene wären z.B.: • Ausgabe, die von „0 %“ bis „100 %“ hochzählt. • Liniengrafik mit z.B. dem Zeichen „|“. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 298 / 409 Lösung siehe Kapitel 14.18. 14.16.3 Aufgabe „Kontaktdaten 3“ Erweitern sie die kleine Kontaktdaten-Verwaltung aus Kapitel 11.5.3. Gegenüber der Aufgabe 11.5.3 soll dieses Programm aber mehrere Arten von Kontakten verwalten können: • Single – entspricht der Person aus der alten Aufgabe. Besteht aus: • Vorname • Nachname • Telefon (als String) • Beschreibung Suchen über Vor- und Nachname • Paar Besteht aus: • Vorname 1 • Vorname 2 • Nachname (gemeinsamer) • Telefon (als String) Suchen über beide Vornamen und den gemeinsamen Nachname • Firmenkontakt • Firmenname • Ansprechpartner • Vorname • Nachname • Position • Telefon (als String) Suchen über Firmenname, Vor- und Nachname des Ansprech-Partners Lösung siehe Kapitel 14.19. 14.16.4 Aufgabe „Kontaktdaten 4“ Trennen sie die Kontaktdaten-Verwaltung aus Kapitel 14.20 in ihre groben Strukturen auf, und packen sie diese in eigene Packages. Besonders wichtig ist hier die vollständige Trennung von Benutzer-Ein- und -Ausgabe und der eigentlichen Programm-Logik. Damit können wir später problemlos eine grafische Oberfläche auf die Programm-Logik aufsetzen. Lösung siehe Kapitel 14.20. 14.16.5 Aufgabe „Tic-Tac-Toe 2“ Bauen sie das Tic-Tac-Toe Programm aus Kapitel 12.8.3 so um, dass Vererbung und © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 299 / 409 Polymorphie benutzt wird. Welche Klasse bieten sich im Tic-Tac-Toe dafür an? Wo können sie das Programm dadurch vereinfachen? Lösung siehe Kapitel 14.21. 14.16.6 Aufgabe „Tic-Tac-Toe 3“ Nach dem Umbau von Kapitel 14.16.5 sollte es möglich sein, dass Tic-Tac-Toe so zu erweitern, dass der Benutzer am Anfang die Spieler dynamisch festlegen kann. D.h. implementieren sie eine Abfrage, mit der der Benutzer definiert, ob Spieler 1 bzw. Spieler 2 jeweils ein Mensch oder der Computer ist. Lösung siehe Kapitel 14.22. 14.16.7 Aufgabe „Tic-Tac-Toe 4“ Implementieren sie einen weiteren einfachen Computer-Spieler, der einfach das Feld per Zufall festlegt. Integrieren sie den Spieler in das Programm und die Spieler-Auswahl von Kapitel 14.16.6. Sehen sie, wie einfach die Erweiterung des Spiels um einen weiteren Spieler geworden ist? Lösung siehe Kapitel 14.23. 14.16.8 Aufgabe „Tic-Tac-Toe 5“ Trennen sie das Tic-Tac-Toe aus Kapitel 14.16.7 in seine groben Strukturen auf, und packen sie diese in eigene Packages. Besonders wichtig ist hier die vollständige Trennung von Benutzer-Ein- und Ausgabe und der eigentlichen Programm-Logik. Immerhin wollen wir ja auch das Tic-Tac-Toe später noch mit einer grafischen Benutzeroberfläche versehen, aber das Spiel nicht neuschreiben müssen sondern die gesamte Logik wiederverwenden können. Lösung siehe Kapitel 14.24. 14.17 Lsg. zu Aufgabe „Obstkorb“ – Kap. 14.16.1 todo... 14.18 Lsg. zu Aufgabe „ProgressBar“ – Kap. 14.16.2 todo... © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 300 / 409 14.19 Lsg. zu Aufgabe „Kontaktdaten 3“ – Kap. 14.16.3 Noch ohne Erläuterungen – todo... import java.io.BufferedReader; import java.io.InputStreamReader; public class Keyboard { private static InputStreamReader isr = new InputStreamReader(System.in); private static BufferedReader reader = new BufferedReader(isr); public static String readString() { try { return reader.readLine(); } catch (Exception x) { } return ""; } } public interface Contact { public boolean find(String pattern); public void input(); public void print(); } public class Single implements Contact { private String forename = ""; private String surename = ""; private String phone = ""; public boolean find(String pattern) { return forename.startsWith(pattern) || surename.startsWith(pattern); } public void input() { System.out.print("Vorname: "); forename = Keyboard.readString(); System.out.print("Nachname: "); surename = Keyboard.readString(); System.out.print("Telefon: "); phone = Keyboard.readString(); } public void print() { System.out.print("- Single: " + forename + " " + surename); if (phone.length()!=0) { System.out.print(" - Tel: " + phone); } System.out.println(); } } public class Pair implements Contact { private private private private String String String String fame1 = ""; fname2 = ""; surename = ""; phone = ""; public boolean find(String pattern) { © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 301 / 409 return fname1.startsWith(pattern) || fname2.startsWith(pattern) || surename.startsWith(pattern); } public void input() { System.out.print("Vorname 1: "); fname1 = Keyboard.readString(); System.out.print("Vorname 2: "); fname2 = Keyboard.readString(); System.out.print("Nachname: "); surename = Keyboard.readString(); System.out.print("Telefon: "); phone = Keyboard.readString(); } public void print() { System.out.print("- Paar: " + fname1 + " & " + fname2 + " " + surename); if (phone.length()!=0) { System.out.print(" - Tel: " + phone); } System.out.println(); } } public class Company implements Contact { private private private private private String String String String String name = ""; fn = ""; sn = ""; position = ""; phone = ""; public boolean find(String p) { return name.startsWith(p) || fn.startsWith(p) || sn.startsWith(p); } public void input() { System.out.print("Firmen-Name: "); name = Keyboard.readString(); System.out.println("Ansprechpartner"); System.out.print("- Vorname: "); fn = Keyboard.readString(); System.out.print("- Nachname: "); sn = Keyboard.readString(); System.out.print("- Position: "); position = Keyboard.readString(); System.out.print("- Telefon: "); phone = Keyboard.readString(); } public void print() { System.out.print("- Firma " + name + " - Ap: " + fn + " " + sn); if (position.length()!=0) { System.out.print(" - Pos: " + position); } if (phone.length()!=0) { System.out.print(" - Tel: " + phone); } System.out.println(); } } import java.util.ArrayList; import java.util.Iterator; public class Contacts { private ArrayList contacts = new ArrayList(); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 302 / 409 public void add(Person p) { contacts.add(p); } public void find(String pattern) { Iterator i = contacts.iterator(); while (i.hasNext()) { Contact c = (Contact)i.next(); if (c.find(pattern)) { c.print(); } } } public void print() { System.out.println("" + contacts.size() + " Kontakte:"); Iterator i = contacts.iterator(); while (i.hasNext()) { Contact c = (Contact)i.next(); c.print(); } } } public class Appl { private Contacts cts = new Contacts(); public void run() { while (true) { System.out.println("Bitte waehlen sie eine Aktion aus"); System.out.println("- l : Liste"); System.out.println("- n : Neu"); System.out.println("- s : Suchen"); System.out.println("- e : Ende"); String in = Keyboard.readString(); System.out.println(); if (in.equalsIgnoreCase("l")) { print(); } else if (in.equalsIgnoreCase("n")) { insert(); } else if (in.equalsIgnoreCase("s")) { find(); } else if (in.equalsIgnoreCase("e")) { return; } System.out.println(); } } private void print() { cts.print(); } private void insert() { Contact c; while (true) { System.out.println("- s : Single"); System.out.println("- p : Paar"); System.out.println("- f : Firma"); System.out.println("- a : Abbruch"); String in = Keyboard.readString(); System.out.println(); if (in.equalsIgnoreCase("s")) { System.out.println("<Eingabe Single>"); c = new Single(); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 303 / 409 break; } else if (in.equalsIgnoreCase("p")) { System.out.println("<Eingabe Paar>"); c = new Pair(); break; } else if (in.equalsIgnoreCase("f")) { System.out.println("<Eingabe Firma>"); c = new Company(); break; } else if (in.equalsIgnoreCase("a")) { return; } } c.input(); cts.add(c); } private void find() { System.out.print("Geben sie einen Suchstring ein: "); String pattern = Keyboard.readString(); cts.find(pattern); } public static void main(String[] args) { Appl appl = new Appl(); appl.run(); } } 14.20 Lsg. zu Aufgabe „Kontaktdaten 4“ – Kap. 14.16.4 todo... 14.21 Lsg. zu Aufgabe „Tic-Tac-Toe 2“ – Kap. 14.16.5 Immer wenn in einer Aufgaben-Stellung von „verschiedenen Arten von irgendwas“ geredet wird, klingt dies nach Vererbung. Und wenn die „verschiedenen Arten von irgendwas“ noch gemeinsam verarbeitet werden müssen, dann ist der Einsatz von Polymorphie meist sinnvoll. 14.21.1 Basis-Klasse „Player“ Im Tic-Tac-Toe aus Kapitel todo gibt es mehrere Arten von Spielern – den Menschen und den Computer, der immer das nächste freie Feld nimmt. Hierfür könnte eine Klassen-Hierarchie also sinnvoll sein. Die erste Aktion ist also die Integration einer Basis-Klasse „Player“. • Da kein Spieler an sich existiert, wird die Basis-Klasse abstrakt gemacht. • Beide Spieler haben die Verwaltung der Spielfarbe gemeinsam, d.h. wird diese in die BasisKlasse verschoben. • Zusätzlich werden die gemeinsamen Funktionen „getName“ und „next“ als abstrakte Funktionen bekannt gemacht. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 304 / 409 public abstract class Player { private int color; public Player(int c) { color = c; } public int getColor() { return color; } public abstract String getName(); public abstract Move next(Board b); } Hier die neue Klassen-Implementierung von „ComputerNext“. Die Klasse „Human“ ändert sich analog. public class ComputerNext extends Player { public ComputerNext(int c) { super(c); } public String getName() { return "Computer (naechstes freies Feld)"; } public Move next(Board b) { for (int i=0; true; i++) { Move m = new Move(i, getColor()); if (b.valid(m)) { return m; } } } } 14.21.2 Haupt-Schleife vereinfachen Werden die Spieler denn irgendwo im Programm gemeinsam verarbeitet? Wenn ja, so könnte dies durch Vererbung und Polymorphie vielleicht vereinfacht werden. Ja, in der Hauptschleife ist zweimal fast der gleiche Code vorhanden, da beide Spieler identisch verarbeitet werden müssen, aber dies bislang nicht mit einem Code-Stück möglich war. Hier noch mal der bisherige Code: public class Appl { public static void main(String[] args) { Board b = new Board(); Human pl1 = new Human(Move.WHITE); ComputerNext pl2 = new ComputerNext(Move.BLACK); b.print(); while (true) { Move m = pl1.next(b); System.out.println(pl1.getName() + " zieht " + m + '\n'); b.set(m); b.print(); © Detlef Wilkening 1997-2016 (*) http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 305 / 409 if (b.wins(pl1.getColor())) { System.out.println(pl1.getName() + " hat gewonnen"); return; } if (b.full()) { System.out.println("Das Spiel ist remis ausgegangen"); return; } (**) m = pl2.next(b); System.out.println(pl2.getName() + " zieht " + m + '\n'); b.set(m); b.print(); if (b.wins(pl2.getColor())) { System.out.println(pl2.getName() + " hat gewonnen"); return; } } } } Das Problem war, dass sich der doppelte Code (von (*) bis (**)) nicht in eine Funktion ziehen liess, da beide Spieler unterschiedlichen Typs waren. void play(Typ pl) { Move m = pl.next(b); ... } // welcher Typ? Human oder ComputerNext Nun gibt es eine gemeinsame Basis-Klasse, und damit ist es möglich eine Funktion für beide Spieler zu schreiben. • Um das Spielende zu propagieren, gibt die „play“ Funktion noch einen boolean Wert zurück. • Außerdem bekommt „play“ neben dem Spieler noch das Board übergeben. • Zusätzlich typisieren wir noch die Spieler „pl1“ und „pl2“ nach „Player“ um – obwoh das hier noch ziemlich egal ist. public class Appl { public static void main(String[] args) { Board b = new Board(); Player pl1 = new Human(Board.WHITE); Player pl2 = new ComputerNext(Board.BLACK); b.print(); while (true) { if (play(b, pl1)) return; if (play(b, pl2)) return; } } public static boolean play(Board b, Player pl) { Move m = pl.next(b); System.out.println(pl.getName() + " zieht " + m + '\n'); b.set(m); b.print(); if (b.wins(pl.getColor())) { System.out.println(pl.getName() + " hat gewonnen"); return true; } if (b.full()) { System.out.println("Das Spiel ist remis ausgegangen"); return true; } © Detlef Wilkening 1997-2016 (*) (**) (***) http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 306 / 409 return false; } } Und warum funktioniert das? 1) Da die Funktionen „getName“ und „next“ auch in Player definiert sind – wenn auch nur abstrakt – compilieren die Zeilen (*), (**) und (***). 2) Dank Polymorphie landen die Funktions-Aufrufe in (*), (**) und (***) auch in den jeweiligen Spieler-Klassen in den richtigen Funktionen. 14.22 Lsg. zu Aufgabe „Tic-Tac-Toe 3“ – Kap. 14.16.6 Das ist jetzt kein wirkliches Problem, da wir in Kapitel 14.21 alle Spieler schon auf „Player“ typisiert haben. public static void main(String[] args) { Board b = new Board(); Player pl1 = new Human(Board.WHITE); Player pl2 = new ComputerNext(Board.BLACK); ... Schon jetzt könnten wir problemlos z.B. dem Computer „weiß“ und dem Menschen „schwarz“ geben: public static void main(String[] args) { Board b = new Board(); Player pl1 = new ComputerNext(Board.WHITE); Player pl2 = new Human(Board.BLACK); ... Man muss dies nur noch durch eine Benutzer-Eingabe dynamisch steuern: import java.io.BufferedReader; import java.io.InputStreamReader; public class Appl { public static void main(String[] args) { Board b = new Board(); Player pl1 = choosePlayer("ersten", Board.WHITE); Player pl2 = choosePlayer("zweiten", Board.BLACK); b.print(); while (true) { if (play(b, pl1)) return; if (play(b, pl2)) return; } } public static Player choosePlayer(String s, int color) { InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); while (true) { System.out.println("Bitte waehlen sie den " + s + " Spieler aus:"); System.out.println("- Mensch (M)"); System.out.println("- Computer erstes freies Feld (E)"); System.out.print("> "); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 307 / 409 try { String in = reader.readLine(); System.out.println(); if (in.compareToIgnoreCase("M") == 0) { return new Human(color); } if (in.compareToIgnoreCase("E") == 0) { return new ComputerNext(color); } } catch (Exception x) { } System.out.println("Fehlerhafte Eingabe!\n"); } } public static boolean play(Board b, Player pl) { ... } } 14.23 Lsg. zu Aufgabe „Tic-Tac-Toe 4“ – Kap. 14.16.7 14.23.1 Klasse „ComputerRandom“ Zuerst implementieren wir einen weiteren Computer-Spieler, der das zu spielende Feld per Zufall auswählt. Damit er problemlos ins Programm integriert werden kann, wird er natürlich von „Player“ abgeleitet. import java.util.Random; public class ComputerRandom extends Player { public ComputerRandom(int c) { super(c); } public String getName() { return "Computer (Zufall)"; } public Move next(Board b) { Random rnd = new Random(); while (true) { int field = rnd.nextInt(9); Move m = new Move(field, getColor()); if (b.valid(m)) { return m; } } } } 14.23.2 Spieler-Auswahl Dann muß er noch ins Menü für die Spieler-Auswahl integriert werden: public static Player choosePlayer(String s, int color) { InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); while (true) { © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 308 / 409 System.out.println("Bitte waehlen sie den " + s + " Spieler aus:"); System.out.println("- Mensch (M)"); System.out.println("- Computer erstes freies Feld (E)"); System.out.println("- Computer Zufall (Z)"); System.out.print("> "); try { String in = reader.readLine(); System.out.println(); if (in.compareToIgnoreCase("M") == 0) { return new Human(color); } if (in.compareToIgnoreCase("E") == 0) { return new ComputerNext(color); } if (in.compareToIgnoreCase("Z") == 0) { return new ComputerRandom(color); } } catch (Exception x) { } System.out.println("Fehlerhafte Eingabe!\n"); } } 14.23.3 Fertig Und das war es. Dank Vererbung und Polymorphie funktioniert das eigentliche Programm auch mit einer Klasse, die zum Zeitpunkt der Programm-Erstellung noch gar nicht bekannt war. Das eigentliche Programm kennt nur Player und bindet die Funktionen dynamisch, d.h. landen die Funktions-Aufrufe an den richtigen Stellen. Nur an den Stellen, an denen Objekte erzeugt werden, muß der Code verändert werden. Und geschickt programmiert, gibt es nur wenige solche Code-Stellen im Programm. 14.24 Lsg. zu Aufgabe „Tic-Tac-Toe 5“ – Kap. 14.16.8 todo... 15 Innere Klassen Seit Java 1.1 können in Java Klassen und Interfaces ineinander verschachtelt werden. Man nennt sie innere Klassen, eingebettete Klassen oder auch verschachtelte Klassen. Es gibt mehrere Sorten von eingebetteten Klassen: 1. Member-Klassen Eingebettete static Klassen (oder auch „statische Member-Klassen“) sind relativ normale Klassen, d.h auch Top-Level Klassen, die logisch einer anderen Klasse zugeordnet sind, und auch auf deren private Elemente zugreifen dürfen. Eingebettete nicht static Klassen (oder auch „Member-Klassen“) sind sogenannte Element-Klassen, deren Instanzen immer einer umgebenden Klassen-Instanz zugeordnet sind. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 309 / 409 Eingebettete Interfaces (oder auch „Member-Interfaces“) sind vergleichbar zu eingebetteten static Klassen, d.h sie sind Top-Level-Interfaces, die logisch einer anderen Klasse zugeordnet sind, und auch auf deren private Elemente zugreifen dürfen. Interfaces sind implizit immer static. 2. Lokale Klassen sind Klassen die innerhalb eines Code-Blocks definiert sind, und auch nur dort benutzt werden können. Obwohl doch etwas anderes, haben sie viele Eigenschaften mit eingebetteten nicht static Klassen gemeinsam. 3. Anonyme Klassen sind lokale Klassen ohne Namen. 15.1 Eingebettete static Klassen Eingebettete static Klassen: • Benötigen den Modifier „static“. • Zugriff über Kombination der Klassen-Namen mit trennendem Punkt. • Verhalten sich wie normale Klassen (Top-Level-Klassen). • Können beliebig tief geschachtelt werden. • Sind logisch der (den) umgebenden Klasse(n) zugeordnet. • Dürfen auch auf private Elemente der umgebenden Klasse(n) zugreifen. • Haben direkten Zugriff auf static Elemente der umgebenden Klasse(n). • Die umgebende Klasse hat auch Zugriff auf private Elemente der inneren Klasse. public class OuterClass { private String pstr = "Private aeussere Element-Variable"; private static String pocv = "Private aeussere Klassen-Variable"; public OuterClass() { System.out.println("Konstruktor OuterClass"); System.out.println(picv); new StaticInnerClass(this); } public static class StaticInnerClass { private static String picv = "Private innere Klassen-Variable"; public StaticInnerClass(OuterClass o) { System.out.println("Konstruktor StaticInnerClass"); System.out.println(o.pstr); System.out.println(pocv); } } } OuterClass o = new OuterClass(); OuterClass.StaticInnerClass i = new OuterClass.StaticInnerClass(o); Ausgabe Konstruktor OuterClass Private Innere Klassen-Variable Konstruktor StaticInnerClass Private aeussere Element-Variable Private aeussere Klassen-Variable © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 310 / 409 Konstruktor StaticInnerClass Private aeussere Element-Variable Private aeussere Klassen-Variable 15.2 Eingebettete nicht static Klassen Eingebettete nicht static Klassen: • Heissen auch Element-Klassen. • Sind logisch der umgebenden Klasse zugeordnet. • Ihre Objekte sind immer mit genau einem Objekt der umgebenden Klasse verbunden. • Dürfen auch auf private Elemente der umgebenden Klasse zugreifen. • Dürfen keine static Elemente enthalten. • Dürfen nicht den Namen einer umgebenden Klasse bzw. Package’s haben. public class OuterClass { private String pstr; public OuterClass(String s) { System.out.println("Konstruktor OuterClass"); pstr = s; new InnerClass(); } public class InnerClass { public InnerClass() { System.out.println("Konstruktor InnerClass"); System.out.println(pstr); } } } OuterClass o = new OuterClass("Call in fct"); Ausgabe Konstruktor OuterClass Konstruktor InnerClass Call in fct Im obigen Beispiel greift der Konstruktor direkt auf eine Objekt-Variable der umgebenden Klasse zu, obwohl hier scheinbar gar kein Objektbezug vorhanden ist. Diesen Objektbezug stellt der Compiler automatisch her. Wird ein Objekt der Elementklasse erzeugt, übergibt der Compiler automatisch die this Referenz des aktuellen Objekts als Objektbezug. Ausserdem erzeugt der Compiler automatisch in der Elementklasse immer ein Attribut für die Referenz auf das zugeordnete umgebenden Klassenobjekt. Wird versucht, eine Elementklasse in einer anderen Klasse zu erzeugen, so meldet der Compiler einen Fehler, da das aktuelle Objekt nicht den richtigen Klassentyp hat. public class A { public void fct() { new OuterClass.InnerClass("Call from A"); } } © Detlef Wilkening 1997-2016 // Error http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 311 / 409 Um ein Objekt der Elementklasse ausserhalb der umgebenden Klasse erzeugen zu können, muss man dem Compiler den impliziten Parameter explizit angeben. Dafür wurde in Java eine spezielle Syntax für den Operator „new“ eingeführt: OuterClass o = new OuterClass("Call from every-wbere"); o.new InnerClass(); Ausgabe Konstruktor OuterClass Konstruktor InnerClass Call from every-where Konstruktor InnerClass Call from every-where Diese new Operator wird wie eine Elementfunktion für das zuzuordnende Objekt aufgerufen. Der Klassenname der Elementklasse muss nicht weiter referenziert werden, da automatisch der Namensraum der umgebenden Klasse benutzt wird. 15.3 Eingebettete Interface’s Eingebettete Interface’s: • sind äquivalent zu eingebetteten static Klassen • sich wie normale Interfaces • können beliebig tief geschachtelt werden • sind logisch der (den) umgebenden Klasse(n) zugeordnet • sind implizit immer static - das Schlüsselwort kann daher weggelassen werden public class OuterClass { public static interface InnerInterface { public void fct(); } } public class Concrete implements OuterClass.InnerInterface { public void fct() { System.out.println("In Concrete.fct()"); } } OuterClass.InnerInterface ii = new Concrete(); ii.fct(); Ausgabe In Concrete.fct() 15.4 Lokale Klassen Eine lokale Klasse wird innerhalb eines Code-Blocks definiert, und ist nur innerhalb desselben bekannt. Im Prinzip ist eine lokale Klasse eine spezielle Version einer eingebetteten Klasse, © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 312 / 409 denn jeder Code-Block existiert in Java innerhalb einer Klasse, und so hat auch eine lokale Klasse einen Klassen- bzw. Objektbezug. Daher können die Regeln für eingebettete Klassen erstmal auf lokale Klassen übertragen werden. • Lokale Klassen sind nur in dem sie definierenden Block bekannt. • Lokale Klassen können nicht als public, protected, private oder static definiert werden. • Lokale Klassen dürfen auch auf private Elemente der umgebenden Klasse zugreifen. • Lokale Klassen dürfen keine static Elemente enthalten. • Lokale Klassen können auf Variablen, Parameter und Ausnahmen im Sichtbarkeit des definierenden Blocks zugreifen, wenn dieses als „final“ definiert sind. • Lokale Klassen können keine Interfaces sein. • Lokale Klassen dürfen nicht den Namen einer umgebenden Klasse bzw. Package’s haben. public class Normal { private String str = "String Attribute in Normal"; public void f { final int i = 42; class LocalClass { public LocalClass() { System.out.println("Konstruktor LocalClass"); System.out.println(str); System.out.println(i); } } LocalClass l = new LocalClass(); } } Normal n = new Normal(); n.f(); Ausgabe Konstruktor LocalClass String Attribute in Normal 42 Hinweis - eine lokale ist gegenüber einer eingebetteten Klasse vorzuziehen, wenn die Klasse nur in einer Funktion benötigt wird. 15.4.1 Externe Verwendung Selbst wenn eine lokale Klasse nur innerhalb des sie definierenden Blocks bekannt ist, so können Objekte der Klasse auch woanders benutzt werden. public interface Interface { public void f(); } public class Normal { public Interface getInterfaceObject { class LocalClass implements Interface { public void f() { © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 313 / 409 System.out.println("LocalClass.f()"); } } return new LocalClass(); } } Normal n = new Normal(); Interface i = n.getInterfaceObject(); i.f(); Ausgabe LocalClass.f() 15.5 Anonyme Klassen Anonyme Klassen sind lokale Klassen ohne Namen. Sie können direkt an der Stelle, wo ein Objekt von ihnen benötigt und erzeugt wird, ohne Namen definiert werden. Im Gegensatz zu der Definition einer lokalen Klasse ist die Definition einer anonymen Klasse ein Ausdruck, und kann daher Teil eines größeren Ausdrucks (z.B. eines Funktionsaufrufs) sein. • Anonyme Klassen haben nur den automatischen Default-Konstruktor. • Objekte anonymen Klassen, die in einer Element-Funktion erzeugt werden, haben automatischen Objekt-Bezug zum aktuellen Objekt der umgebenden Klasse – d.h. dem Objekt, für das die Element-Funktion aufgerufen wurde (dem „this“ der Element-Funktion) – siehe Beispiel. • Objekte anonymen Klassen, die in einer Klassen-Funktion erzeugt werden, haben keinen Objekt-Bezug. Syntax new Basisklasse () { Klassendefinition } public interface Inter { public void f(); } public class Appl { private String s = "Attribute in Appl"; public static void fct(Inter i) { System.out.println("Appl.fct bekommt Objekt vom Typ Inter uebergeben"); i.f(); } public static void main(String[] args) { fct(new Inter() { public void f() { System.out.println("- f() von anonymer Klasse"); //System.out.println("- - Attribut-Zugriff: \"" + s + "\""); Fehler System.out.println("- - kein Attribut-Zugriff"); } }); Appl appl = new Appl(); appl.doit(); } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 314 / 409 public void doit() { fct(new Inter() { public void f() { System.out.println("- f() von anonymer Klasse"); System.out.println("- - Attribut-Zugriff: \"" + s + "\""); } }); } } Ausgabe Appl.fct bekommt Objekt vom Typ Inter uebergeben - f() von anonymer Klasse - - kein Attribut-Zugriff Appl.fct bekommt Objekt vom Typ Inter uebergeben - f() von anonymer Klasse - - Attribut-Zugriff: "Attribute in Appl" Hinweis – anonyme Klassen mögen seltsam und wie ein Spezialfall wirken, aber sie sind z.B. als Implementierung von Listener-Interfaces bzw. -Klassen beim Event-Handling in Swing das tägliche Brot – siehe z.B. Kapitel todo. 15.6 Virtuelle Maschine Die virtuellen Java Maschine kennt keine eingebetteten Klassen. Damit sie trotzdem den erzeugten Byte Code verarbeiten kann, wendet der Compiler einen kleinen Trick an: er erzeugt statt der eingebetteten Klassen scheinbare Top-Level Klassen, deren Namen sich aus dem Namen der umgebenden Klasse, einem $ Zeichen und dem Namen der eingebetteten Klasse ergibt. public class OuterClass { public static class StaticInnerClass { } } So finden sich nach einem solchen Code-Stück im entsprechenden Byte-Code Verzeichnis die Dateien: • OuterClass.class • OuterClass$StaticInnerClass.class wieder. Schauen sie sich das Ausgabe-Verzeichnis ruhig mal an. 16 GUI Programmierung mit Swing Bei vielen Programmen wird heutzutage eine grafische Bedienoberfläche mit Fenstern, Menüs, Maus, uwm. erwartet. Für die Programmierung grafischer Oberflächen enthält Java die Bibliothek „Swing“, die Klassen für Fenster, Buttons und vieles mehr enthält. Swing kapselt die Unterschiede zwischen den einzelnen Betriebssystemen, so dass eine mit Java erstellt Anwendung auf allen Plattformen läuft, auf denen eine virtuelle Maschine JVM © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 315 / 409 vorhanden ist. Dies macht Swing u.a. dadurch, dass es das komplette GUI selber zeichnet und sich nicht auf die nativen Widgets des jeweiligen Betriebssystems abstützt. Dieses Vorgehen hat mehrere Konsequenzen: • Vorteile: o Alle Swing Elemente stehen immer auf jeder JVM zur Verfügung, auch wenn das zugrunde liegende OS gar keine entsprechenden Widgets kennt. o Das Look&Feel von Swing Anwendungen kann jederzeit ausgetauscht werden, und dies sogar zur Laufzeit. So ist z.B. auf einem Windows Rechner ein Motif Look&Feel möglich, oder umgekehrt – siehe auch Kapitel 16.6. • Nachteile: o Bestimmte betriebssystem-spezifische Features sind nicht in Swing vorhanden, da sie nur bedingt auf andere Plattformen abbildbar sind - z.B. die neuen Windows 7 Elemente69. Aufgrund der vollständigen Kapselung der unter der JVM liegenden Plattform (OS und Prozessor) besteht ausser via JNI70 auch keine Möglichkeit solche Features anzusprechen. o Swing GUIs sind etwas langsamer und speicherfressender als native Widgets – dies ist aber in der Praxis meistens nicht relevant. Die Klassen-Bibliotheken für die grafische Oberfläche haben sich in der Geschichte von Java mehrmals geändert. Beim Umstieg vom JDK 1.0 zum JDK 1.1 wurde u.a. das Event-Modell komplett umgekrempelt. Mit dem JDK 1.2 wurden die alten AWT (Abstract Window Toolkit) Klassen um die neueren viel leistungsfähigeren JFC (Java Foundation Classes) Klassen erweitert, die ein Bestandteil von Swing sind und die alten AWT Klassen als Basis-Klassen beinhalten. Im gesamten Tutorial wird mit den Swing-Klassen gearbeitet und auf die alte AWT Bibliotheken, soweit sie nicht noch als Basis von Swing relevant sind, gar nicht mehr eingegangen. Dieses Kapitel stellt nur eine erste Einführung in die Programmierung von grafischen Oberflächen dar, indem es das berühmte „Hallo Welt“ implementiert. In den folgenden Kapiteln werden einzelne Bereiche der Programmierung von grafischen Oberflächen weiter vertieft. Hinweis – die Swing Packages wurden mit dem JDK 1.2 gegenüber Vorversionen umbenannt. Wer also mit Quelltexten aus der JDK 1.1.x Phase konfrontiert wird, dem werden veraltete import Anweisungen begegnen: • Früher „com.sun.java.swing“ • Jetzt: „javax.swing“ Lange Zeit war mein Standard Beispiel die Unterstützung von Tray-Icons, aber seit dem JDK 1.6 werden nun auch diese unterstützt. Hier sehen Sie damit auch sehr schön, warum die Java Bibliothek so gross sein muß: da die OS-API nicht direkt zur Verfügung steht, müssen alle Fähigkeiten des OS in der Java Bibliothek abgebildet werden. 70 JNI (Java Native Interface) ist eine Schnittstelle in Java, mit der C Funktionen angesprochen werden können. 69 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 316 / 409 Bemerkung – mittlerweile gibt es mehrere alternative GUI-Toolkits für Java, relevant davon sind aber eigentlich nur folgende beiden: • Eins davon ist SWT, das im Rahmen des Eclipse Projekts entwickelt wurde. Im Gegensatz zu Swing baut es auf die native Widgets des jeweiligen Betriebssystems auf. Dadurch ist das Look&Feel von SWT besser auf das OS abgestimmt und SWT ist auch performanter. Auf der anderen Seite macht man sich politisch aber von einem Toolkit abhängig, das nicht fester Bestandteil von Java ist, und eben daher nicht automatisch auf allen Plattformen mit einer JVM verfügbar ist. Außerdem ist SWT nicht so gut erweiter- und anpaßbar wie Swing. Dafür bietet es mit dem RCP (Rich-Client-Plattfom) Framework eine mächtige und sehr hilfreiche Basis für Applikations-Entwicklung. • Seit neustem exisitiert auch eine Java Anbindung der C++ GUI Klassen-Bibliothek QT, die z.B. als C++ Version das GUI-Toolkit für den KDE Desktop unter Linux darstellt. Achtung – die Kapitel über GUIs mit Swing sind natürlich nur eine kleine Einführung. Das Thema GUI Entwicklung ist viel zu umfangreich, als das es hier umfassend behandelt werden könnte - allein mit Swing kann man ganze Bücher füllen71. Diese Einführung versucht die wichtigsten Konzepte von GUI Programmierung und Swing vorzustellen 16.1 Ein einfaches Fenster Wenn es nur darum geht, ein Fenster auf den Bildschirm zu zaubern, hält sich der Aufwand in Grenzen – wir haben das Programm schon in Kapitel 3.1.2 kennen gelernt. import javax.swing.JFrame; public class Appl { public static void main(String[] args) { JFrame frame = new JFrame("Mein GUI Fenster 1"); frame.setLocation(20, 20); frame.setSize(600, 400); frame.setVisible(true); } } In „main“ wird ein Objekt der Swing-Frame Klasse „JFrame“ erzeugt, die für ein einfaches Fenster darstellt – dem Konstruktor wird der Fenstertitel übergeben. Mit ‘setLocation’ wird der Startort, mit ‘setSize’ die Grösse, und mit ‘setVisible’ die Sichtbarkeit gesetzt. Fertig. Aber diese Minimal-Lösung hat mehrere Nachteile: • Mit Schliessen des Fensters wird nicht mal das Programm beendet - sie können es nur auf der Kommandozeile mit [Strg]+[C] abschiessen. • Das JFrame-Klasse liefert nur ein allgemeines Default-Verhalten, das eigentlich nie reicht. In dem Fenster soll ja schließlich was passieren. Um mit dem Schliessen des Fensters das Programm zu beenden, muss für das Fenster ein 71 Die es auch gibt – dicke Wälzer nur über Swing. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 317 / 409 entsprechendes Flag gesetzt werden – dies geschieht mit „setDefaultCloseOperation“ und der Konstanten „JFrame.EXIT_ON_CLOSE“. import javax.swing.JFrame; public class Appl { public static void main(String[] args) { JFrame frame = new JFrame("Mein GUI Fenster 2"); frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); frame.setLocation(20, 20); frame.setSize(600, 400); frame.setVisible(true); } } In einem späteren Kapitel über Event-Handling werden wir das Beenden des Programms beim Schließen des Fensters anders lösen, indem wir uns selber in die Event-Bearbeitung einhängen. Aber das wird nur ein Beispiel zum Verständnis des Event-Handlings in Swing sein – der normale Weg sollte der hier beschriebene mit dem Setzen der Default-Close-Operation sein. 16.2 Ein etwas besseres Fenster Besser wäre also ein eigenes Fenster, das wir nach unseren Vorstellungen formen (programmieren) können - wir würden das Default-Verhalten von JFrame aber gerne beibehalten. Was macht man dann in Java? Natürlich erben! Hier ein zweites Programm, das zwar nicht mehr kann als das vorherige, aber schon mal die Vorbereitungen für mehr darstellt. public class Appl { public static void main(String[] args) { MyFrame frame = new MyFrame("Mein eigenes Fenster"); frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); frame.setLocation(20, 20); frame.setSize(600, 400); frame.setVisible(true); } } import javax.swing.JFrame; public class MyFrame extends JFrame { public MyFrame(String title) { super(title); } } Im Prinzip ist es das alte Programm – nur wird statt der Swing Klasse „JFrame“ eine eigene Frame-Klasse „MyFrame“ benutzt, die sich von „JFrame“ ableitet. Da „MyFrame“ keine neue Funktionalität hinzufügt, verhält sich das neue Programm wie das alte. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 318 / 409 16.3 Ein grafisches „Hallo Welt“ 16.3.1 Das normale grafische „Hallo Welt“ Um ein grafisches „Hallo Welt“ zu erhalten, müssen wir im Fenster auch „Hallo Welt“ ausgeben. Wie macht man das? • Immer, wenn ein Fenster neu gezeichnet werden muss, wird automatisch die Funktion „paint“ aufgerufen, die daher für eigene Ausgaben überschrieben werden muss – siehe weiter unten und Kapitel 17. • Die Funktion „paint“ bekommt ein „Graphics“ Objekt übergeben72, das u.a. Funktionen für die Textausgabe in einem Fenster zur Verfügung stellt – siehe Kapitel 16.4. import java.awt.Graphics; import javax.swing.JFrame; public class MyFrame extends JFrame { public MyFrame(String title) { super(title); } public void paint(Graphics g) { super.paint(g); g.drawString("Hallo Welt", 50, 50); } } Für die Ausgabe überschreiben wir die Funktion „paint“, die an den Koordinaten „50/50“ den Text „Hallo Welt“ ausgibt - mehr dazu in Kapitel 17 und Kapitel 16.4. Abb. 16-1 : Ein grafisches „Hallo Welt“ Programm Achtung - es ist ganz wichtig - nicht nur für die Funktion „paint“ sondern auch für viele weitere Funktionen, die wir zukünftig kennenlernen werden - dass zuerst die überschriebene Funktion der Basis-Klasse (hier „paint“) aufgerufen wird. Das Neu-Zeichnen unseres Fensters besteht ja nicht nur aus der Ausgabe von „Hallo Welt“, sondern noch aus anderen Dingen - und die geschehen natürlich in den Basisklassen, und sollten nicht verhindert werden. Und man sollte zuerst die Super-Funktion aufrufen, da sie ansonsten möglicherweise alle unsere Aktionen wieder zerstört. Genau genommen wird ein „java.awt.Graphics2D“ übergeben – aber diese Details ignorieren wir erstmal. 72 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 319 / 409 Hinweis – warum funktioniert das? Hier ist natürlich wieder Vererbung und Polymorphie im Spiel. Der interne Fenster-Verwaltungs-Mechanismus der JVM kennt unsere Fenster Klasse „MyFrame“ natürlich nicht, aber er arbeitet auf einer Basisklasse von „MyFrame“. In dieser ist die Funktion „paint“ definiert, und die wird intern aufgerufen. Wir haben diese Funktion aber nun überschrieben, und dank Polymorphie landet damit der Aufruf von „paint“ in „MyFrame.paint“. 16.3.2 Probleme mit dem JDK 1.5 und zum Teil auch mit dem JDK 1.6 Das im vorherigen Kapitel 16.3.1 beschriebene Java-Programm funktioniert mit dem JDK 1.5 und manchen JDK 1.6 Versionen nicht fehlerfrei73 - zumindest in den von mir getesten Versionen 1.5.0_04 bis 1.5.0_09. Mit allen möglichen alten JDKs, z.B. 1.2, 1.3, 1.4, 1.4.1, 1.4.2 und auch mit einigen JDK 1.6 Versionen funktioniert das Programm korrekt. Was ist denn hier nun das Problem? Ganz einfach – immer wenn das Fenster in irgendeiner Form vergrößert wird (d.h. es wird mit der Maus oder der Tastatur größer gezogen, oder es wird maximiert), dann wird der Inhalt des Fensters nicht neu gezeichnet, d.h. die Paint-Funktion wird nicht aufgerufen. In der realen Praxis ist dies kein Problem, da dort eigentlich nie direkt das Haupt-Frame für Paint-Aktionen genutzt wird, sondern im Haupt-Frame eigentlich immer weitere GUI-Elemente (z.B. Menüs, Buttons, Panels, usw.) liegen. Und bei denen klappt das alles auch z.B. im JDK 1.5 problemlos. Problematisch ist der Bug aber für dieses Tutorial und auch andere Lehrbücher. Denn hier sollen die Beispiele natürlich möglichst einfach sein, weshalb im Frame oft keine weiteren Elemente liegen – und damit tritt der Fehler auf. In der Praxis sollten Sie sich dieses Problems bewußt sein – und die Musterlösungen aus dem Tutorial entsprechend verändern. Eine mögliche Lösung z.B. ist das Frame mit einem eigenen Panel (siehe Kapitel 20.8) zu füllen, und alle Aktionen statt im Frame im Panel unterzubringen. Als Beispiel hier das „Hallo-Welt“ Programm in einer Version mit zusätzlichem Panel, die u.a. auch mit dem JDK 1.5 funktioniert. import javax.swing.JFrame; public class Appl { public static void main(String[] args) { MyFrame frame = new MyFrame("Hallo-Welt Fenster fuer das JDK 1.5"); frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); frame.setLocation(20, 20); frame.setSize(600, 400); frame.setVisible(true); } Möglicherweise gibt es – wenn Sie dieses Tutorial lesen – eine neuere JDK Version. Und möglicherweise ist dort der Bug nicht mehr vorhanden. 73 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 320 / 409 } import javax.swing.JFrame; public class MyFrame extends JFrame { public MyFrame(String title) { super(title); getContentPane().add(new MyPanel()); } } import java.awt.Graphics; import javax.swing.JPanel; public class MyPanel extends JPanel { public void paint(Graphics g) { super.paint(g); g.drawString("Hallo Welt", 50, 50); } } 16.4 Graphics Objekt Die Klasse „java.awt.Graphics“ stellt die Schnittstelle eines Swing Programms für die Ausgabe auf dem Bildschirm dar. Die Klasse bietet verschiedenste Element-Funktionen an um Pixel, Linien, Kreise, Rechtecke, Texte, uvm. auf dem Bildschirm auszugeben. Lassen sie sich von Eclipse einfach mal die Menge an Funktionen anzeigen, bzw. schauen sie in die Java-Hilfe. Bei vielen Funktionen sagt der Name intuitiv was die Funktion macht, und auch die Parameter benötigen kaum Erklärung, von daher sollten einfache Anwendungen kein Problem sein. Beispiele von Element-Funktionen in „java.awt.Graphics“ Funktion void drawLine(int x1, int y1, int x2, int y2) void drawRect(int x, int y, int width, int height) void fillRect(int x, int y, int width, int height) void drawOval(int x, int y, int width, int height) void fillOval(int x, int y, int width, int height) void setColor(Color c) void setPaintMode() void setXORMode(Color c1) Beschreibung Zeichnet eine Linie Zeichnet ein Rechteck Zeichnet ein ausgefülltes Rechteck Zeichnet eine Elipse Zeichnet eine ausgefüllte Elipse Setzt die Zeichen-Farbe Setzt den Zeichen-Mode auf Überschreiben Setzt den Zeichen-Mode auf XOR zur übergebenen Farbe import java.awt.Color; import java.awt.Graphics; import javax.swing.JFrame; public class MyFrame extends JFrame { © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 321 / 409 public MyFrame(String title) { super(title); setSize(380, 340); } public void paint(Graphics g) { super.paint(g); g.drawString("Hallo Welt", 20, 60); g.fillOval(100, 100, 80, 50); g.setColor(Color.RED); g.drawRoundRect(40, 200, 300, 40, 20, 20); g.setColor(Color.BLUE); g.drawLine(50, 150, 300, 300); } } Abb. 16-2 : Ein GUI Programm mit bunter Ausgabe im Fenster Hinweis - während des normalen Programm-Ablaufs, z.B. bei einer Benutzer-Interaktion oder während einer Berechnung, kann es ohne weiteres notwendig sein direkt auf den Bildschirm zu zeichnen. Auch in diesen Fällen benötigt man ein Graphics Objekt - man kann es sich für jedes Swing Objekt mit der Funktion „getGraphics()“ holen. Eine Anwendung dafür findet sich z.B. im Scribble Programm in Kapitel 18.2.4. Achtung – das Überschreiben der Paint-Funktion des Haupt-Fensters führt mit dem JDK 1.5 zu Problemen – siehe Kapitel 16.3.2. 16.5 Klasse „Color“ Im letzten Beispiel wurde u.a. die Zeichen-Farbe für das Graphics Objekt auf „rot“ bzw. „blau“ gesetzt - siehe Kapitel 16.4. Für Farben gibt es die Klasse „java.awt.Color“. Sie enthält u.a. Konstanten für die wichtigsten Farben wie „schwarz – BLACK“, „weiß – WHITE“, „rot – RED“, „blau – BLUE“, „grün – GREEN“, usw. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 322 / 409 Objekte der Klasse „Color“ können aber z.B. auch durch die gewünschten RGB Werte (rot, grün, blau) erzeugt werden – die Klasse enthält hierfür verschiedene Konstruktoren. Color c = new Color(128, 0, 128); Die Graphics Klasse enthält die Funktion „setColor“ zum Setzen der Zeichen-Farbe. Alle Swing Klassen haben die Funktionen „setForeground“ und „setBackgroundColor“, um die Vorder- und Hintergrund-Farbe zu setzen – die Vordergrund-Farbe entspricht der Zeichenfarbe. 16.6 Änderung der Oberfläche in den Windows-Style Alle GUI Programme in diesem Tutorial werden mit dem normalen Swing-Style erzeugt, d.h. der Oberflächen-Skin ist Swing. Nehmen wir z.B. das GUI-Fenster aus Kapitel 19.4, das obwohl der Screenshot unter Windows XP mit klassischem Windows Skin erzeugt wurde – nicht wirklich nach Windows aussieht: Abb. 16-3 : GUI-Fenster aus Kapitel 19.4 mit Swing-Style Möglicherweise möchten Sie aber, dass Ihre Programme unter Windows auch einen WindowsStyle haben. Das können Sie in Swing erreichen, indem Sie in Ihrem Programm den Style entsprechend setzen. Hier der Quelltext aus Kapitel 19.4 mit verändertem Windows-Style: import javax.swing.JFrame; import javax.swing.UIManager; public class Appl { public static void main(String[] args) { try { UIManager.setLookAndFeel( "com.sun.java.swing.plaf.windows.WindowsLookAndFeel"); } catch (Exception e) { © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 323 / 409 // Klappt es wohl nicht - auch egal - dann eben mit Swing-Style... } MyFrame frame = new MyFrame(); frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); frame.setLocation(20, 20); frame.setSize(600, 400); frame.setVisible(true); } } import java.awt.BorderLayout; import java.awt.Container; import java.awt.GridLayout; import javax.swing.JButton; import javax.swing.JFrame; import javax.swing.JPanel; @SuppressWarnings("serial") public class MyFrame extends JFrame { public MyFrame() { super("Fenster mit verschachtelten Layouts"); setSize(400, 300); JPanel panel = new JPanel(new GridLayout(3, 2)); panel.add(new JButton("1")); panel.add(new JButton("2")); panel.add(new JButton("3")); panel.add(new JButton("4")); panel.add(new JButton("5")); panel.add(new JButton("6")); Container c = getContentPane(); c.setLayout(new BorderLayout()); c.add(new JButton("North"), BorderLayout.NORTH); c.add(new JButton("West"), BorderLayout.WEST); c.add(new JButton("East"), BorderLayout.EAST); c.add(new JButton("South"), BorderLayout.SOUTH); c.add(panel, BorderLayout.CENTER); } } Abb. 16-4 : GUI-Fenster aus Kapitel 19.4 mit Windows-Style © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 324 / 409 16.7 Aufgaben 16.7.1 Aufgabe „Schwebende Kugel“ Implementieren sie ein Fenster, dass eine schwebende Kugel wie in der Abb. 16-5 enthält. Hinweis – es sind wirklich nur die problemlosesen Zeichen-Funktionen aus Kapitel 16.4 benutzt worden. Es ist einfach ein bisschen geschickt mit der Farbe gespielt worden. Abb. 16-5 : Fenster mit schwebender Kugel Lösung siehe Kapitel 16.8. 16.7.2 Aufgabe „Farb-Fenster“ Implementieren sie ein Fenster, dessen Inhalt viele kleine Quadrate sind, die jeweils eine andere Farbe haben. Überlegen sie, wie sie die drei Farb-Dimensionen „RGB“ auf eine zweidimensionale Fläche abbilden, bzw. wie die den Farb-Raum sinnvoll reduzieren. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 325 / 409 Lösung siehe Kapitel 16.9. 16.8 Lsg. zu Aufgabe „Schwebende Kugel“ – Kap. 16.7.1 todo... 16.9 Lsg. zu Aufgabe „Farb-Fenster“ – Kap. 16.7.2 todo... 17 Philosophie der GUI Programmierung Bevor wir weiter gehen, müssen einige Dinge über die Philosophie von GUI Programmierung gesagt werden. Diese Dinge sind ganz unabhängig von Swing oder Java, sondern gelten wohl für alle GUI Bibliotheken. • Der Bildschirm gehört nicht dem Programm oder einem Fenster, sondern dem Betriebssystem. • Der Kontrollfluß wird vom GUI vorgegeben und ist stark interaktiv. 17.1 Besitzer des Bildschirms Im Gegensatz zu alten DOS oder CPM Zeiten unterstützen heute alle modernen Betriebssysteme die Möglichkeit mehrere Programme gleichzeitig auszuführen74. Wenn dann diese Programme noch GUI Anwendung sind, wie die meisten Anwendungen heute, hat das einige Konsequenzen für den Zugriff auf den Bildschirm: • Ein Programm kann nicht einfach beliebig auf den Bildschirm schreiben, da es nicht selbstverständlich ist, dass der Bildschirm ihm gehört, und nicht ein anderes Programm dort seine Ausgaben macht. • Ein Programm kann seinen Ausgabe-Zustand nicht auf dem Bildschirm abfragen, da mittlerweile ein anderes Programm dort seine Ausgaben gemacht haben kann. • Ein Programm muss immer in der Lage sein, seine Darstellung zu erneuern, da der Benutzer die Fenster beliebig nach vorne, hinten, seitwärts und sonstwas verschieben kann, d.h. ein bislang nicht sichtbarer Teil des Fensters oben liegt und neu gezeichnet werden muss - siehe z.B. Kapitel 18.2.4 und Kapitel todo. Die letzten beiden Punkte bedingen, dass ein Programm immer ein komplettes Modell75 seiner Ausgabe enthalten muss. Damit kann ein Programm dann jederzeit seine Darstellung auf dem Bildschirm erneuern, wenn es dazu aufgefordert wird. Dies geschieht via OS, JVM und PaintDas heisst, sie sind multitasking fähig. Mit Modell sind hier alle Daten und deren Beziehungen gemeint, die das Programm visualisiert. 74 75 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 326 / 409 Funktion. 17.2 Kontrollfluß Klassische Programme wurden oft nach dem EVA Prinzip76 aufgebaut - hierbei ist der Programmfluss relativ einfach - ein Schritt folgt nach dem anderen. Abb. 17-1 : Aufbau klassischer Programme Auch komplexere klassische Programme haben eine klare Kontrollflußsteuerung in ihrer Programmlogik. Ein Beispiel dafür ist unser „Tic-Tac-Toe“, dessen „main“ den Kontrollfluß klar vorgibt – siehe Kapitel 12.11.1. In einem modernen grafischen Benutzerinterface werden viel höhere Anforderungen an ein Programm gestellt. Der Benutzer kann beliebige Aktionen durch die Tasten, Menü, Buttons,... auswählen, er kann das Fenster vergrössern, verkleinern, in den Hintergrund legen bzw. nach vorne holen. Andere Programme können die Umgebung verändern, der Benutzer kann neue Einstellungen vornehmen - und auf all das muss das Programm jederzeit sauber und kontrolliert reagieren. Damit dies mit einem halbwegs vernünftigen und überschaubaren Programmieraufwand möglich ist, hat sich das Programmiermodell von EVA hin zu einem Event-gesteuertem Modell verändert. Hierbei liegt die Kontrolle nicht mehr primär bei dem Programm, sondern beim Betriebssystem, das bei jedem Ereigniss (Event) eine entsprechende Funktion des Programms aufruft – sogenannte „Callback-Funktionen“. Abb. 17-2 : Aufbau Event-gesteuerter Programme Dieses event-gesteuerte Programmiermodell hat sich in der Praxis sehr bewährt, und findet sich natürlich auch in Java wieder. An jedes mögliche Ereignis kann der Programmierer eine Funktion ankoppeln, die bei ihrem Auftritt automatisch ausgeführt wird. Das ganze ist am Anfang recht ungewohnt, und es sind viele Fragen zu klären: • Wie kann auf Events reagiert werden, d.h. wie wird eine Callback-Funktion in Java implementiert? • Wie können Events verhindert oder modifiziert werden können? • Wie werden Events bei mehreren aufeinanderliegenden grafischen Elementen propagiert? • Können Events programmgesteuert ausgelöst werden? • Welche Elemente können Events auslösen, und welche darauf reagieren? • Wie können Events und die dahinter liegende Benutzerlogik von den Daten und der Programmlogik getrennt werden? 76 Eingabe, Verarbeitung, Ausgabe © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 327 / 409 Diese und weitere Fragen müssen von einem Event-Modell beantwortet werden - und dieses Modell gibt damit den Rahmen - oder auch die Philosophie - vor, in dem ein Programmierer ein Programm Design erschaffen und implementieren kann und muss. Hinweis – damit haben wir natürlich ein Problem mit unserer Tic-Tac-Toe Lösung aus Kapitel 14.24 bzw. auch den darauf aufbauenden Lösungen. In dieser Lösung liegt die Steuerung des Kontrollflußes klar auf der Seite der Spiele-Logik, während uns das GUI Programmier-Modell diese Option nicht läßt. In einem späteren Kapitel werden wir mögliche Lösungen dieses Konflikts diskutieren. 18 Event-Modelle von Swing Swing kennt zwei Event-Modelle: • das interne Event-Modell – siehe Kapitel 18.2, und • das externe Event-Modell – siehe Kapitel 18.3. Achtung – nicht Java kennt zwei Event-Modelle, sondern Swing. Die Sprache stellt AusdrucksElemente zur Verfügung, in deren Rahmen Bibliotheken und Programme entwickelt werden können. Wie eine Bibliothek (wie z.B. Swing) dann ihr Funktionalität dem Benutzer zur Verfügung stellt – d.h. wie sie benutzt wird – ist eine Frage der Bibliothek und nicht der Sprache. Die Sprache gibt den Rahmen vor indem sie sich bewegen kann. Hinweis – in Java, genau genommen in den GUI Bibliotheken von Java, hat sich die Philosophie des Event-Handlings, d.h. das Event-Modell, mehrmals geändert. Die EventModelle der JDKs 1.0, 1.1 und 1.2 unterscheiden sich zum Teil sehr stark voneinander. Dies spiegelt sich weniger in der Sprache wieder77, sondern in erster Linie in den sehr unterschiedlichen GUI Bibliotheken der JDK Versionen. 18.1 Events und Gruppierungen Ein unerfahrener Programmierer mag sich die Frage stellen: „Welche Events gibt es überhaupt?“ Grundsätzlich ist ein Event eine elementare Aktion des Benutzers, die eine Reaktion des Programms zur Folge haben kann. Dazu gehören z.B.: • Jede Art der Tastatur-Betätigung. • Jede Maus-Aktion wie Linksklick, Rechtsklick, Linksdoppelklick, Maus bewegen, usw. Da das Event-Modell eine Eigenschaft einer Bibliothek ist, sollte es sich eigentlich in keiner Weise in der Sprache wiederspiegeln. Aber das JDK 1.1 ist u.a. um die inneren Klassen – siehe Kapitel 15 – erweitert worden, um das neue externe Event-Modell eleganter programmieren zu können. 77 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 328 / 409 • Fenster-Aktionen wie Fenster wird bewegt, vergrößert, verkleinert, minimiert, maximiert, geschlossen, usw. • Fokus-Aktionen, d.h. ein Element verliert bzw. bekommt den Fokus. • uvm. Die exakt vorhandenen Events sind vom jeweiligen grafischen Element abhängig, das den Fokus hat. Z.B. hat ein Baum-Element sicher Events für das Selektieren von Einträgen, für das Auf- und Zuklappen von Teilbäumen, uvm. Diese Events machen aber z.B. für ein einfaches Fenster gar keinen Sinn. Da alle grafische Elemente viele Events haben, sind diese in Swing gruppiert. Gerade komplexere Elemente wie Tabellen und Bäume wären sonst sehr unübersichtlich. So gibt es z.B. bei den meisten GUI Elementen drei Gruppen von Maus-Events: • Mouse • MouseMotion • MouseWheel Hinweis – es kann auch programm-interne Events geben, z.B. Timer-Events – siehe Kapitel 20.10 – aber die meisten Events werden durch externe Aktionen des Benutzers angestoßen, wie Tastatur- oder Maus-Aktionen. Um die Sache einfach zu halten, gehen wir im folgenden davon aus, dass alle Events von außen angestoßen werden. Ist dies nicht der Fall, durchläuft das Event nur nicht soviele Stufen – die grundsätzliche Abarbeitung in Swing ist aber immer gleich. 18.2 Internes Event-Modell 18.2.1 Der Fluß eines Events Jedes Event, das der Benutzer auslöst – z.B. das Klicken mit der linken Maus-Taste in ein Fenster – durchläuft mehrere Stufen: • Das Betriebssystem bildet dabei die unterste Ebene. Es interagiert mit Hilfe von Treibern mit der Hardware, und wird von jeder Aktion unterrichtet. Das BS interpretiert die Aktion, und leitet sie gegebenenfalls an das aktive Programm weiter – in unserem Fall die JVM. • Die JVM kennt das aktuelle Fenster – genau genommen das grafische Element, das den Fokus hält – und ruft eine entsprechende Event-Behandlungs-Funktion auf. • Das aktive grafische Element hat sich beim Erzeugen automatisch bei der JVM registriert – dies passiert automatisch in den Konstruktoren der Basisklassen - und hat die entsprechende Event-Behandlungs-Funktion möglicherweise überschrieben. So erfährt es von dem Event und kann darauf reagieren. • Hierbei wird zuerst die Funktion „processEvent“ aufgerufen, die das Event dann auf die jeweiligen Event-Gruppen „process“ Funktionen verteilt - siehe Kapitel 18.2.2. Abb. 18-1 : Der Fluß eines Events © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 329 / 409 18.2.2 Swings „process“ Funktionen und Event-Klassen Für jede Gruppe gibt es in den Swing Klassen „process“ Funktionen, die bei den jeweiligen Events aufgerufen werden. Hier ein Teil der „process“ Funktionen der Fenster Klasse „JFrame“78: • protected void processWindowEvent(WindowEvent e) • protected void processWindowFocusEvent(WindowEvent e) • protected void processWindowStateEvent(WindowEvent e) • protected void processKeyEvent(KeyEvent e) • protected void processMouseEvent(MouseEvent e) • protected void processMouseMotionEvent(MouseEvent e) • protected void processMouseWheelEvent(MouseWheelEvent e) Alle diese Funktionen sind „protected“, da sie nicht von außen aufgerufen werden sollen, aber natürlich zum Überschreiben gedacht sind, d.h. in den abgeleiteten Klassen ansprechbar sein müssen. Alle „process“ Funktionen bekommen ein Event-Objekt als Parameter, das detailierte Informationen zum jeweiligen Event enthält. Welche Informationen das jeweils sind, ist vom jeweiligen abhängig. Die Namen der Event-Klassen entsprechen häufig dem dem der jeweiligen „process“ Funktion ohne Präfix „process“, d.h. die Funktion „processKeyEvent“ z.B. bekommt einen Parameter vom Typ „KeyEvent“. Die Event-Klassen sind in den verschiedensten Packages definiert. Aber die grundlegenden Event-Klassen wie „WindowEvent“, „KeyEvent“ oder „MouseEvent“ kommen aus „java.awt.event“. Achtung – im Normallfall sollte eine überschriebene „process“ Funktion zuerst die überschriebene Basis-Klassen Funktion aufrufen. Geschieht dies nicht, schneiden sie die geerbte Event-Behandlung vom Event-Fluß ab – d.h. sie entfällt komplett. Im Normallfall werden sie dies nicht erreichen wollen, ganz im Gegenteil. Achtung - ein Teil der „process“ Funktionen - wie z.B. „processMouseMotionEvent“ - müssen explizit aktiviert werden. Der Grund ist, dass manche dieser Events sehr häufig passieren können - z.B. Bewegungen der Maus. Je nach Event-Gruppe, Plattform und Anwendung könnte diese aufwändige Event-Verarbeitung zu einer spürbaren PerformanceVerschlechterung führen. Auf modernen Hardware-Plattformen sollte dies zwar nicht der Fall sein, aber beim Design von Swing wurde hier auf „Nummer Sicher“ gegangen. „process“ Funktionen können explizit mit der Funktion „enableEvents“ und einer passenden Die Klasse „JFrame“ hat - obwohl aus Sicht des GUIs eine recht einfache Klasse - immerhin schon 14 „process“ Funktionen (Stand JDK 1.4.2). 78 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 330 / 409 Bitmaske an- und ausgestellt werden - siehe z.B. Kapitel 18.2.4. Implizit werden sie auch mit der Registrierung eines entsprechenden Listener-Objekts aktiviert - siehe Kapitel 18.3.2. 18.2.3 Beispiel „WindowClose“ In Kapitel 16.1 haben wir gelernt wie man ein erreicht, dass das Schließen eines „JFrame“ automatisch auch das Programm beendet. Selbst wenn die dort dargestellt Methode sicher für die meisten Fälle ausreichend und sinnvoll ist, wollen wir hier als Beispiel dies mal selber implementieren. Dazu muß man wissen, dass das Schließen eines Fensters unter die normalen Window-Events fällt. • Daraus folgt, dass wir in unserer eigenen Frame-Klasse die Funktion „processWindowEvent“ überschreiben müssen. • Dann wollen wir auf den Event-Typ „WindowEvent.WINDOW_CLOSING“ reagieren, der mit „getID()“ abgefragt werden kann. • In diesem Fall wollen wir das Programm direkt beenden - dies geht mit „System.exit(0)“. • Und auch hier gilt - wie in Kapitel 18.2.2 und Kapitel 16.3 erklärt - dass natürlich zuerst die Basis-Klassen Funktion „processWindowEvent“ aufgerufen werden sollte. Alles zusammen ergibt das dann folgendes Programm: public class Appl { public static void main(String[] args) { MyFrame frame = new MyFrame("Fenster mit eigenem Programm-Ende"); // Nicht mehr notwendig - darum kuemmern wir uns jetzt selbst // frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); frame.setLocation(20, 20); frame.setSize(600, 400); frame.setVisible(true); } } import java.awt.event.*; import javax.swing.*; public class MyFrame extends JFrame { public MyFrame(String title) { super(title); } protected void processWindowEvent(WindowEvent e) { super.processWindowEvent(e); if (e.getID() == WindowEvent.WINDOW_CLOSING) { System.exit(0); } } } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 331 / 409 18.2.4 Beispiel „Scribble“ Um das interne Event-Modell näher zu verstehen, wollen wir eine erste Version eines „Scribble“ Programms mit Hilfe des internen Event-Modells implementieren. Scribble ist quasi ein extrem einfaches Zeichen-Programm, bei dem sie mit der Maus Kurven und Linien in das Fenster zeichnen können. • Mit dem Drücken einer beliebigen Maus-Taste beginnt das Zeichnen. • Solange die Maus-Taste gedrückt ist, wird entsprechend den Maus-Bewegungen gezeichnet. • Mit dem Loslassen der Maus-Taste endet die Zeichen-Aktion. Abb. 18-2 : Ein einfaches Scribble Zeichen-Programm Als erstes müssen wir uns überlegen, wie das Programm prinzipiell arbeiten soll: • Die gezeichneten Striche sind Geraden zwischen den Punkten, die für die Maus-Bewegung als Event gemeldet werden. • Für den ersten Strich benötigen wir also die x/y Koordinate der Maus beim Maus-Klick und nach der ersten Maus-Bewegung. Die x/y Koordinaten der Maus beim Maus-Klick müssen also gespeichert werden - hierzu werden zwei Attribute „lastX“ und „lastY“ in der FensterKlasse definiert. • Für alle weiteren Striche werden die x/y Koordinaten zwischen Ende des letzten Strichs und der jeweils nächsten Maus-Bewegung benötigt - auch hier benutzen wir zur Speicherung die Attribute „lastX“ und „lastY“. • Das Loslassen der Maus-Taste ist kein relevantes Event, da es an einer Maus-Position passiert, die schon als Maus-Bewegung gemeldet wurde. • Damit auch nur ein einzelner Maus-Klick gezeichnet wird, muss noch der erste Punkt explizit gezeichnet werden. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 332 / 409 Folgende Events sind also für Scribble wichtig: • Eine beliebige Maus-Taste wurde gedrückt - dies fällt unter die normalen Maus-Events, d.h. die Funktion „processMouseEvent“ muss überschrieben werden. Das relevante Event ist das mit der ID „MouseEvent.MOUSE_PRESSED“ - die x/y Koordinaten können mit „getX“ und „getY“ erfragt werden. • Das Ziehen mit gedrückter Maus-Taste muss verfolgt werden - dies ist eine Drag-Aktion und fällt unter die Maus-Motion-Events, d.h. die Funktion „processMouseMotionEvent“ muss überschrieben werden. Das relevante Event ist hier das mit der ID „MouseEvent.MOUSE_DRAGGED“. Und so ergibt sich folgender Quelltext für unser „Scribble 1“. Denken sie daran, dass beide Event-Gruppen explizit mit „enableEvents“ aktiviert werden müssen - siehe auch Kapitel 18.2.2. import javax.swing.JFrame; public class Appl { public static void main(String[] args) { JFrame frame = new ScribbleFrame("Scribble 1 : internes Event-Modell"); frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); frame.setLocation(20, 20); frame.setSize(600, 400); frame.setVisible(true); } } import java.awt.*; import java.awt.event.*; import javax.swing.*; public class ScribbleFrame extends JFrame { private int lastX; private int lastY; public ScribbleFrame(String title) { super(title); enableEvents( AWTEvent.MOUSE_EVENT_MASK | AWTEvent.MOUSE_MOTION_EVENT_MASK); } protected void processMouseEvent(MouseEvent e) { super.processMouseEvent(e); if (e.getID() == MouseEvent.MOUSE_PRESSED) { onMousePressed(e); } } protected void processMouseMotionEvent(MouseEvent e) { super.processMouseMotionEvent(e); if (e.getID() == MouseEvent.MOUSE_DRAGGED) { onMouseDragged(e); } } private void onMousePressed(MouseEvent e) { lastX = e.getX(); lastY = e.getY(); Graphics g = getGraphics(); g.drawLine(lastX, lastY, lastX, lastY); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 333 / 409 } private void onMouseDragged(MouseEvent e) { int x = e.getX(); int y = e.getY(); Graphics g = getGraphics(); g.drawLine(lastX, lastY, x, y); lastX = x; lastY = y; } } Hiemit haben sie ein ganz einfaches Scribble programmiert, das aber genau unter der in Kapitel 17.1 beschriebenen Problematik leidet: „Der Bildschirminhalt kann und wird auf Anforderung nicht neu gezeichnet, d.h. er geht unter Umständen verloren.“ Probieren sie es aus: Minimieren sie z.B. das Fenster, schieben sie es in den Hintergrund, oder verschieben sie es zum Teil aus dem Bildschirmbereich, und stellen sie es dann wieder her. Der Bildschirm ist dann leer. Das Fenster muss neu gezeichnet werden, und zum Teil wird es das auch, denn um Rahmen, Titelleiste, usw. kümmern sich die Swing Basis-Klassen. Aber unser Inhalt fehlt, da wir uns nicht darum kümmern. Wir werden dieses Problem in Kapitel 18.5 lösen, aber vorher schauen wir uns das externe Event-Modell an. 18.3 Externes Event-Modell Das externe Event-Modell baut auf das sogenannte Observer Pattern auf. Von daher macht es Sinn, das Oberver-Pattern kurz für sich zu besprechen. „Design Pattern“ bzw. Entwurfsmuster79 stellen mehr oder weniger fest umrissene DesignEntwürfe für konkrete Problem-Situationen dar. Sie enthalten keine exakte Code-Vorgabe, sondern sie erklären das konzeptionelle Klassen-Gerüst und die Verteilung der Aufgaben, um ein Problem zu lösen. Normalerweise finden sie d.h. keinen fertigen Quelltext, der nur noch abgetippt werden muß, wie z. B. bei Algorithmen, sondern sie müssen selbständig den Entwurf in passenden Code für ihr Projekt überführen. 18.3.1 Observer Pattern 18.3.1.1 Problem Ein häufiges Problem ist, dass parallel mehrere (oftt unterschiedliche) Sichten auf einen DatenBestand existieren. Wird der Daten-Bestand nun über eine der Sichten geändert, sollen sich alle anderen mit aktualisieren. Entwurfsmuster sind ein wichtiges Thema in der Software-Entwicklung. Die Bibel der Entwurfsmuster ist das sehr empfehlenswerte Buch „Entwurfsmuster“ von Gamma, Helm, Vlissides und Johnson. 79 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 334 / 409 Ein sehr großes Beispiel wäre eine Daten-Visualisierung, bei der die Daten sowohl in Form einer Tabelle als auch in Form verschiedener Grafiken angezeigt werden. Werden nun die Daten in einer Sicht manipuliert, so sollen sich die anderen Sichten parallel anpassen Ein anderes – vielleicht aktuelleres Beispiel – ist Eclipse. Änderungen im Quelltext werden simultan im Package-Explorer, im Outline und in anderen Sichten angezeigt. Das grundsätzliche Problem „ich-bin-interessiert-an-Änderungen-der-Daten“ ist ein StandardProblem der Software-Entwicklung und exisitiert in allen Größen-Ordnungen. Aber wie löst man es verständlich, wartbar, übersichtlich, erweiterbar, wiederverwendbar, usw. in einer objektorientierten Sprache? Die Antwort ist das Observer-Pattern. 18.3.1.2 Lösung • • • • Allgemeine abstrakte Basis-Klasse für die Daten Allgemeines Interfaces für die Sichten Kopplung nur zwischen der Basis-Klasse und dem Interface Benachrichtigungs-Mechanismus wird nur einmal in der Daten-Basis-Klasse implementiert, die konkreten Daten erben ihn. Abb. 18-3 : Klassen-Diagramm Observer-Pattern Abb. 18-4 : Interaktion Observer-Pattern 18.3.1.3 Betrachtung der Abhängigkeiten der Klassen todo... 18.3.1.4 Umsetzung in Java todo... Abb. 18-5 Klassen-Diagramm Observer-Pattern in Java import java.util.ArrayList; import java.util.Iterator; public class Observable { private ArrayList observers = new ArrayList(); public void addObserver(Observer obs) { observers.add(obs); } public void notifyObservers() { © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 335 / 409 Iterator i = observers.iterator(); while (i.hasNext()) { Observer obs = (Observer)i.next(); obs.update(); } } } public class Data extends Observable { private String date = ""; public void setDate(String arg) { date=arg; super.notifyObservers(); } public String getDate() { return date; } } public interface Observer { public void update(); } import java.io.*; public class View1 implements Observer { private Data data; public View1(Data d) { data = d; data.addObserver(this); } public void update() { System.out.println("View 1 Daten \"" + data.getDate() + '"'); } public void editData() { System.out.print("View 1 - bitte geben sie neue Daten ein:\n> "); try { InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); String in = reader.readLine(); data.setDate(in); } catch (Exception x) { } } } import java.io.*; public class View2 implements Observer { private Data data; public View2(Data d) { data = d; data.addObserver(this); } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 336 / 409 public void update() { System.out.println("View 2 Daten \"" + data.getDate() + '"'); } public void editData() { System.out.print("View 2 - bitte geben sie neue Daten ein:\n> "); try { InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); String in = reader.readLine(); data.setDate(in); } catch (Exception x) { } } } public class Appl { public static void main(String[] args) { Data data = new Data(); View1 view1 = new View1(data); View2 view2 = new View2(data); data.setDate("Aus main"); view1.editData(); view2.editData(); } } mögliche Ausgabe View 1 Daten "Aus main" View 2 Daten "Aus main" View 1 - bitte geben sie neue Daten ein: > Dateneingabe via View1 View 1 Daten "Dateneingabe via View1" View 2 Daten "Dateneingabe via View1" View 2 - bitte geben sie neue Daten ein: > Und was aus View 2 View 1 Daten "Und was aus View 2" View 2 Daten "Und was aus View 2" Hinweis – da das Oberver-Pattern ein Standard-Problem in der Software-Entwicklung ist, gibt es schon seit dem JDK 1.0 in der Java Bibliothek die Klasse „java.util.Observer“ und das Interface „java.util.Observable“. Für viele Probleme sind diese Klassen allemal ausreichend. 18.3.2 Listener-Interfaces Das externe Event-Modell ist im Prinzip die konsequente Anwendung des Observer Patterns auf Events. Jedes Event wird quasi als Daten-Änderung eines Observables betrachtet und an alle registrierten Observer gesendet. Für jede Event-Gruppe – siehe Kapitel todo – gibt es ein entsprechendes Listener-Interface (das entspricht quasi dem Observer-Interface im Observer-Pattern). Und für jedes Event der Event-Gruppe gibt es eine abstrakte Funktion, die überschrieben werden muss (sie entsprechen quasi jeweils der „update“ Funktion des Observer-Interfaces. Hier beispielhaft die Definitionen der Listener-Interfaces für die Maus- und Maus-Motion-Events © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 337 / 409 aus dem Package „java.awt.event“: package java.awt.event; public interface MouseListener implements EventListener { public public public public public void void void void void mouseClicked(MouseEvent e); mousePressed(MouseEvent e); mouseReleased(MouseEvent e); mouseEntered(MouseEvent e); mouseExited(MouseEvent e); } package java.awt.event; public interface MouseMotionListener implements EventListener { public void mouseDragged(MouseEvent e); public void mouseMoved(MouseEvent e); } Jedes GUI-Element, dass eine Event-Gruppe unterstützt, bietet entsprechende Funktionen an, um Listener beim GUI-Element zu registrieren und wieder abzumelden. So hat die Klasse „JFrame“ u.a. die Funktionen: • public void addMouseListener(MouseListener l) • public void removeMouseListener(MouseListener l) • public void addMouseMotionListener(MouseMotionListener l) • public void removeMouseMotionListener(MouseMotionListener l) Hinweis – die Listener-Interfaces heißen immer wie die Event-Gruppen mit Postfix „Listener“. Die zugehörigen „add“ und „remove“ Funktionen heißen immer wie die Listener-Interfaces mit Präfix „add“ bzw. „remove“. Wir könnten unser Scribble also auch folgendermassen implementieren: todo... import java.awt.*; import java.awt.event.*; import javax.swing.*; public class ScribbleFrame extends JFrame implements MouseListener, MouseMotionListener { private int lastX; private int lastY; public ScribbleFrame(String title) { super(title); addMouseListener(this); addMouseMotionListener(this); } private void onMousePressed(MouseEvent e) { lastX = e.getX(); lastY = e.getY(); Graphics g = getGraphics(); g.drawLine(lastX, lastY, lastX, lastY); } private void onMouseDragged(MouseEvent e) { © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 338 / 409 int x = e.getX(); int y = e.getY(); Graphics g = getGraphics(); g.drawLine(lastX, lastY, x, y); lastX = x; lastY = y; } public void mouseClicked(MouseEvent e) { } public void mouseEntered(MouseEvent e) { } public void mouseExited(MouseEvent e) { } public void mousePressed(MouseEvent e) { onMousePressed(e); } public void mouseReleased(MouseEvent e) { } public void mouseDragged(MouseEvent e) { onMouseDragged(e); } public void mouseMoved(MouseEvent e) { } } 18.3.3 Listener-Adapter-Klassen Die notwendige Implementierung aller Funktionen der Listener-Interfaces ist häufig nervig, denn in der Praxis werden viele Funktionen leer überschrieben, da das Event für die Anwendung nicht wichig ist – siehe z.B. Quelltext „Scribble2“ in Kapitel 18.3.2. Eine Alternative ist hier die Verwendung der Listener-Adapter-Klassen statt der ListenerInterfaces. Die Adapter-Klassen implementieren alle Funktionen des jeweiligen Interfaces mit leeren Funktionen. Hier beispielhaft die Definitionen der Listener-Adapter-Klassen für die Maus- und Maus-MotionEvents aus dem Package „java.awt.event“: package java.awt.event; public class MouseAdapter implements MouseListener { public public public public public void void void void void mouseClicked(MouseEvent e) {} mousePressed(MouseEvent e) {} mouseReleased(MouseEvent e) {} mouseEntered(MouseEvent e) {} mouseExited(MouseEvent e) {} } package java.awt.event; public class MouseMotionAdapter implements MouseMotionListener { public void mouseDragged(MouseEvent e) {} © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 339 / 409 public void mouseMoved(MouseEvent e) {} } Das es überhaupt Interfaces gibt liegt daran, dass Java keine Mehrfach-Vererbung von Klassen kennt, und daher z.B. das „Scribble2“ sich nicht von „JFrame“, „MouseAdapter“ und „MouseMotionAdapter“ ableiten kann sondern nur von einer Klasse – siehe Kapitel todo. Dem gegenüber kann eine Klasse beliebig viele Interfaces implementieren, und kann so mehrere Aufgaben übernehmen – siehe auch Kapitel todo. Hinweis – die Listener-Adapter-Klassen heißen immer wie die Event-Gruppen mit Postfix „Adapter“. 18.3.4 Innere Klassen Das „Scribble2“ in Kapitel todo ist erstmal kein wirklicher Fortschritt gegenüber dem „Scribble1“ in Kapitel todo – eher im Gegenteil, da auch die unbenutzten Interface-Funktionen leer implementiert werden müssen. Aber das externe Event-Modell hat gegenüber dem internen einen wichtigen Vorteil: • Im internen Event-Modell muss man immer eine eigene Klasse schreiben, die sich von der normalen Klasse ableitet und die entsprechende „process“ Funktion überschreibt. Eine solche eigene abgeleitete Klasse ist aber spezifisch für das aktuelle Problem und könnte daher nicht wiederverwendet werden. Daher benötigt man dann viele abgeleitete Klassen mit sehr ähnlichem Code, was sehr unschön ist. • Im externen Event-Modell dagegen können sie einen Observer beim zu beobachtenden Element registrieren, ohne das sie dieses anpassen müßten. Um diesen Unterschied zwischen den beiden Event-Modellen noch mal klar zu machen, implementieren wir mit beiden Strategien die gleiche Aufgabe: • Ausgabe der Maus-Koordinaten auf der Console bei Drücken einer beliebigen Maus-Taste in einem Fenster. • D.h. konkret soll beim Drücken einer beliebigen Maus-Taste in einem Fenster die möglichst „private“ Element-Funktion „out“ der Klasse „Appl“ mit den x- und y-Koordinaten der Maus aufgerufen werden. • Die Element-Funktion „out“ der Klasse „Appl“ soll die Koordinaten dann mit einer Meldung auf der Console ausgeben. Bei der Aufgabe geht es darum, dass ein Event (hier Maus-Taste drücken) eines GUI-Elements (hier das Fenster) für ein Objekt einer anderen Klasse (hier „Appl“) eine Aktion auslöst (d.h. eine Element-Funktion aufruft). Dies ist eine Standard-Aufgabe. Denken sie z.B. an einen Button in einem Fenster. Wird der Button betätigt soll im Fenster irgendwas passieren. Die eigentliche Aktion ist also sicher nicht Bestandteil des Buttons, sondern des Fensters. Das Fenster in unserer Aufgabe entspricht hier also dem Button, das „Appl“ Objekt dem Fenster, und die „out“ Funktion der Aktion. Und da die Aktion ein Teil des Verhaltens des Gesamt© Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 340 / 409 Moduls ist, sollte „out“ möglichst „private“ sein. Hinweis – die Aufgabe ist absichtlich so gewählt, da sie mit unserem aktuellen Wissen problemlos implementierbar ist, denn wir kennen die Fenster-Klasse „JFrame“ und die MausEvents, aber z.B. noch keine Buttons. 18.3.4.1 Implementierung mit dem internen Event-Modell Um die Aufgabe mit dem internen Event-Modell zu lösen, muss: • eine eigene Fenster-Klasse von „JFrame“ abgeleitet werden, • die Maus-Events im Konstruktor enabled werden, und • die entsprechende „process“ Funktion überladen werden. Außerdem muß die Element-Funktion „out“ des „Appl“ Objekts aufgerufen werden können dazu muß das „Appl“ in unserer Fenster-Klasse bekannt sein - d.h. implementieren wir ein entsprechendes Attribut „appl“ und setzen dies im Konstruktor, der nun natürlich auch einen solchen Parameter benötigt. Die Funktion „out“ kann hierbei leider nicht „private“ sein. Der Rest ist Pflicht, und sollte kein Problem mehr sein. import javax.swing.JFrame; public class Appl { public static void main(String[] args) { Appl appl = new Appl(); appl.run(); } private void run() { MyFrame frame = new MyFrame(this); frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); frame.setLocation(20, 20); frame.setSize(600, 400); frame.setVisible(true); } public void out(int x, int y) { System.out.println("Maus-Pressed an " + x + "/" + y); } } import java.awt.AWTEvent; import java.awt.event.MouseEvent; import javax.swing.JFrame; public class MyFrame extends JFrame { private Appl appl; public MyFrame(Appl a) { super("Maus-Pressed Koordinaten auf Console via internem Event-Modell"); appl = a; enableEvents(AWTEvent.MOUSE_EVENT_MASK); } protected void processMouseEvent(MouseEvent e) { super.processMouseEvent(e); if (e.getID() == MouseEvent.MOUSE_PRESSED) { © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 341 / 409 appl.out(e.getX(), e.getY()); } } } 18.3.4.2 Implementierung mit dem externen Event-Modell Die Implementierung mit dem externen Event-Modell ist hier ganz einfach. • Da „Appl“ keine Basis-Klasse hat80, können wir „Appl“ direkt von „MouseAdapter“ ableiten, und sind nicht auf das Interface „MouseListener“ angewiesen. Darum brauchen wir auch nur die „process“ Funktion überladen, die uns interessieren – siehe Kapitel todo. • Da wir das externe Event-Modell benutzen und keine speziellen Fenster-Fähigkeiten brauchen, können wir direkt die Fenster-Klasse „JFrame“ nehmen. • Jetzt können wir die „out“ Funktion auch „private“ machen – siehe Aufgaben-Stellung. import java.awt.event.MouseAdapter; import java.awt.event.MouseEvent; import javax.swing.JFrame; public class Appl extends MouseAdapter { public static void main(String[] args) { Appl appl = new Appl(); appl.run(); } private void run() { JFrame frame = new JFrame("M-P Ko auf Console via externem Event-Modell"); frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); frame.setLocation(20, 20); frame.setSize(600, 400); frame.addMouseListener(this); frame.setVisible(true); } public void mousePressed(MouseEvent e) { out(e.getX(), e.getY()); } private void out(int x, int y) { System.out.println("Maus-Pressed an " + x + "/" + y); } } Die Implementierung mit dem externen Event-Modell ist hier so einfach da: • „Appl“ keine Basis-Klasse hat. • Wir nur für ein GUI-Element „mousePressed“ überschreiben müssen. 18.3.4.3 Implementierung mit einer non-static Member-Klasse Was wäre aber, wenn es zwei Fenster gäbe, für die die „MousePressed“ Events überwacht werden müssen? Das ginge natürlich auch, aber dann müßte die Funktion „mousePressed“ Jedenfalls keine explizite von uns. Natürlich ist „Appl“ implizit von „java.lang.Object“ abgeleitet. 80 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 342 / 409 unterscheiden, welches Fenster das Event abgetriggert hat. Prinzipiell funktioniert das, da die Event-Klassen (hier „MouseEvent“) u.a. auch Informationen über das Source-Element enthalten – die Frage, die sich stellt, ist: „Will man das?“ Übertragen auf ein Fenster mit vielen Buttons hieße das z.B., dass die „mousePressed“ Funktion alle Buttons kennen und unterscheiden können muß. Das ergibt weder schönen noch wirklich wartbaren Code. Besser wäre es sicher, pro Button eine eigene kleine unabhängige „process“ Funktion zu haben. Das Problem ist doch eigentlich, dass man folgendes braucht: • Eine von „MouseListener“ abgeleitete „kleine“ Klasse. • Am besten eine spezielle Klasse ohne explizite Basis-Klasse, damit man „MouseAdapter“ benutzen kann, und sie sich nur um die Event-Gruppe kümmern muss. • Ein Objekt dieser Klasse muss mit dem Objekt der übergeordneten Klasse verbunden sein – in unserem Beispiel z.B. mit dem „Appl“ Objekt, oder mit dem Fenster-Objekt im Button Beispiel. • Und es wäre schön, wenn sie auf die „private“ Elemente des übergeordneten Objekts zugreifen könnte – z.B. auf „private“ Funktionen – um das Modul-Konzept der übergeordneten Klasse nicht aufzubrechen. Erinnern sie sich? So was gibt es, und nennt sich „eingebette nicht-static Klasse“, „innere nichtstatic Klasse“, „Member-Klasse“ oder auch „Element-Klasse“ – siehe Kapitel 15.2. Schreiben wir unser Beispiel mal auf eine Member-Klasse um. import java.awt.event.MouseAdapter; import java.awt.event.MouseEvent; import javax.swing.JFrame; public class Appl { class MouseEventTarget extends MouseAdapter { public void mousePressed(MouseEvent e) { out(e.getX(), e.getY()); } } public static void main(String[] args) { Appl appl = new Appl(); appl.run(); } private void run() { JFrame frame = new JFrame("M-P Koor. auf Console mit Member-Klasse"); frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); frame.setLocation(20, 20); frame.setSize(600, 400); frame.addMouseListener(new MouseEventTarget()); frame.setVisible(true); } private void out(int x, int y) { System.out.println("Maus-Pressed an " + x + "/" + y); } } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 343 / 409 18.3.4.4 Implementierung mit einer lokalen Klasse Da die Klasse nur an einer Stelle benötigt wird, können wir sie auch zu einer lokalen Klasse machen – siehe Kapitel 15.4. import java.awt.event.MouseAdapter; import java.awt.event.MouseEvent; import javax.swing.JFrame; public class Appl { public static void main(String[] args) { Appl appl = new Appl(); appl.run(); } private void run() { JFrame frame = new JFrame("M-P Koor. auf Console mit lokaler Klasse"); frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); frame.setLocation(20, 20); frame.setSize(600, 400); class MouseEventTarget extends MouseAdapter { public void mousePressed(MouseEvent e) { out(e.getX(), e.getY()); } } frame.addMouseListener(new MouseEventTarget()); frame.setVisible(true); } private void out(int x, int y) { System.out.println("Maus-Pressed an " + x + "/" + y); } } 18.3.4.5 Implementierung mit einer anonymen Klasse Und da auch nur ein Objekt der Klasse benötigt wird, kann man daraus auch eine anonyme Klasse machen – siehe Kapitel 15.5. import java.awt.event.MouseAdapter; import java.awt.event.MouseEvent; import javax.swing.JFrame; public class Appl { public static void main(String[] args) { Appl appl = new Appl(); appl.run(); } private void run() { JFrame frame = new JFrame("M-P Koor. auf Console mit anonymer Klasse"); frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); frame.setLocation(20, 20); frame.setSize(600, 400); frame.addMouseListener(new MouseAdapter() { public void mousePressed(MouseEvent e) { out(e.getX(), e.getY()); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 344 / 409 } }); frame.setVisible(true); } private void out(int x, int y) { System.out.println("Maus-Pressed an " + x + "/" + y); } } Die Verwendung einer anonymen Klasse – abgeleitet von einer Event-Adapter-Klasse – als Event-Listener-Klasse ist das normale Verfahren zum Event-Handling in Swing. 18.3.5 Scribble 3 mit anonymen Listener-Klassen Lassen sie uns jetzt unser Scribble mit dem externen Event-Modell und anonymen ListenerKlassen implementieren. import javax.swing.JFrame; public class Appl { public static void main(String[] args) { JFrame frame = new ScribbleFrame("Scrib 3 mit anonymen Listener-Klassen"); frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); frame.setLocation(20, 20); frame.setSize(600, 400); frame.setVisible(true); } } import java.awt.*; import java.awt.event.*; import javax.swing.*; public class ScribbleFrame extends JFrame { private int lastX; private int lastY; public ScribbleFrame(String title) { super(title); addMouseListener(new MouseAdapter() { public void mousePressed(MouseEvent e) { onMousePressed(e); } }); addMouseMotionListener(new MouseMotionAdapter() { public void mouseDragged(MouseEvent e) { onMouseDragged(e); } }); } private void onMousePressed(MouseEvent e) { lastX = e.getX(); lastY = e.getY(); Graphics g = getGraphics(); g.drawLine(lastX, lastY, lastX, lastY); } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 345 / 409 private void onMouseDragged(MouseEvent e) { int x = e.getX(); int y = e.getY(); Graphics g = getGraphics(); g.drawLine(lastX, lastY, x, y); lastX = x; lastY = y; } } 18.4 Vergleich der Event-Modelle Wann nimmt man nun welches Event-Modell? • Im Normallfall ist das externe Event-Modell vorzuziehen. Im Zusammenspiel mit anonymen Listener-Klassen ist es einfach und unproblematisch zu verwenden. • Bei der Entwicklung eigener bzw. spezialisierter GUI-Elemente kann es Sinn machen, sich in die interne Event-Verarbeitung einzuklinken. Auf dieser Ebene hat man mehr viel Einfluss und Möglichkeiten, der bei der Entwicklung eigener Elemente notwendig sein kann. Aber man kann auch viel kaputt machen, da ein Eingriff in die interne Event-Verarbeitung eine Operation am offenen Herzen des Elements ist. Man sollte daher wissen, was man macht und was man will. Ansonsten sollte man besser die Finger davon lassen. 18.5 Scribble 4 mit Daten-Modell Bleibt zum Schluß noch eine Sache zu tun: die versprochene Implementierung81 eines Scribbles, das auch nach dem Neu-Zeichnen sein Bild darstellt – siehe Kapitel 17.1. Im Prinzip ist das ganz einfach: wir müssen uns das gezeichnete Bild merken, und es auf Abruf (d.h. in der Funktion „paint“ – siehe Kapitel 16.3) zeichnen. Dieses Problem zwingt uns dazu, uns Gedanken zu machen, was denn unser Scribble-Bild eigentlich ist bzw. wie es aufgebaut ist: • Jede x/y-Koordinate ist ein Punkt. • Mit der Maus zeichnet der Benutzer Linien-Züge, d.h. eine Aneinander-Reihung (oder Verkettung) von Linien – sogenannte „Polygone“. Hierbei ist der Extrem-Fall zu beachten, dass ein Polygon auch nur aus einem einzelnen Punkt bestehen kann. Ein Polygon ist also eine Reihe von beliebig vielen, aber mindestens einem Punkt. • Von diesen Polygonen kann das Scribble beliebig viele enthalten. Aus dieser Problem-Analyse ergeben sich zwangsläufig folgende notwendige Klassen: • Eine Klasse für Punkte, die x/y- Koordinaten aufnehmen kann. Hiermit sind wir schnell fertig, da es im Package „java.awt“ eine einfache Klasse „Point“ für Punkte gibt. • Eine Klasse „Polygon“ für Polygone, die beliebig viele Punkte aufnehmen kann. Um mindestens einen Punkt zu garantieren, wird der erste Punkt direkt im Konstruktor übergeben. 81 Siehe Ende Kapitel 18.2.4. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 346 / 409 • Und eine Klasse „PaintModel“ für das gesamte Bild, die beliebig viele Polygone aufnehmen kann. Die Klasse heißt „PaintModel“ und nicht „Picture“, da man in der SoftwareEntwicklung häufig von Modellen redet, wenn strukturierte Daten gemeint sind. Z.B. werden wir bei Swing-Tabellen noch sogenannte „TableModel‘s“ kennen lernen – siehe Kapitel todo. Damit wäre Schritt 1 unserer Klassen-Entwicklungs-Überlegungen – siehe Kapitel 11.2 – erledigt. 1. Überlegung - Art der Objekte 2. Überlegung - Schnittstelle 3. Überlegung - Vererbung 4. Überlegung - Implementierung Weitere Überlegungen – todo... import import import import java.awt.Graphics; java.awt.Point; java.util.ArrayList; java.util.Iterator; public class Polygon { private ArrayList points = new ArrayList(); // Ersten Punkt doppelt einfuegen, damit mindestens // zwei Punkte da sind - siehe Funktion 'paint'. public Polygon(Point p) { points.add(p); points.add(p); } public void addPoint(Point p) { points.add(p); } // Diese Funktion setzt voraus, dass mindestens zwei Punkte // vorhanden sind - der Konstruktor garantiert dies. // - waere kein Punkt da => Exception in Zeile (*) // - waere nur ein Punkt da => kein Zeichnen, da die Schleife // nie betreten wird. public void paint(Graphics g) { Iterator it = points.iterator(); Point start = (Point) it.next(); // (*) while (it.hasNext()) { Point end = (Point) it.next(); g.drawLine(start.x, start.y, end.x, end.y); start = end; } } } import import import import java.awt.Graphics; java.awt.Point; java.util.ArrayList; java.util.Iterator; public class PaintModel { private ArrayList polygons = new ArrayList(); private Polygon actuellPolygon; © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 347 / 409 public void addPolygon(Point p) { actuellPolygon = new Polygon(p); polygons.add(actuellPolygon); } public void addPoint(Point p) { actuellPolygon.addPoint(p); } public void paint(Graphics g) { Iterator it = polygons.iterator(); while (it.hasNext()) { Polygon poly = (Polygon) it.next(); poly.paint(g); } } } import java.awt.*; import java.awt.event.*; import javax.swing.*; public class ScribbleFrame extends JFrame { private int lastX; private int lastY; private PaintModel model = new PaintModel(); public ScribbleFrame(String title) { super(title); addMouseListener(new MouseAdapter() { public void mousePressed(MouseEvent e) { onMousePressed(e); } }); addMouseMotionListener(new MouseMotionAdapter() { public void mouseDragged(MouseEvent e) { onMouseDragged(e); } }); } private void onMousePressed(MouseEvent e) { lastX = e.getX(); lastY = e.getY(); Graphics g = getGraphics(); g.drawLine(lastX, lastY, lastX, lastY); model.addPolygon(new Point(lastX, lastY)); } private void onMouseDragged(MouseEvent e) { int x = e.getX(); int y = e.getY(); Graphics g = getGraphics(); g.drawLine(lastX, lastY, x, y); lastX = x; lastY = y; model.addPoint(new Point(x, y)); } public void paint(Graphics g) { super.paint(g); model.paint(g); } } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 348 / 409 18.6 Aufgaben 18.6.1 Aufgabe „MainFrame“ Schreiben sie eine eigene Frame-Klasse „MainFrame“, die von „JFrame“ abgeleitet ist, aber beim Schliessen des Fensters immer automatisch das Programm beendet. • Welche Möglichkeiten haben sie diese Aufgabe zu lösen? • Welche Vor- und Nachteile haben die einzelnen Lösungen? • Welche würden sie bevorzugen? 18.6.2 Aufgabe „Scribble 5 zeichnet Rechtecke“ Implementieren sie mit dem externen Event-Modell ein Rechteck-Scribble, dass statt LinienZüge Rechtecke zeichnet. • Entwickeln sie zuerst eine Version ohne Modell, und konzentrieren sie sich auf die EventBehandlung und die eigentliche Zeichen-Aufgabe. • Danach entwickeln sie ein passendes Modell, und integrieren es in ihr Rechteck-Scribble. 19 Swing Layouts Für viele GUI-Elemente gibt es in Swing natürlich fertige Klassen, z.B. für verschiedene Arten von Buttons, für Labels, Eingabe-Felder, Listen, Tabellen, Bäume, uvm. Ein Teil dieser Klassen besprechen wir in Kapitel 20. Bevor wir diese Elemente besprechen, sollten wir aber wissen, wie man sie in ein Fenster einfügt und positioniert. Das ganze hat was von der Henne/Ei-Problematik. Man kann keine Elemente in ein Fenster einfügen, wenn man keine Layouts kennt. Aber wie soll man Layouts erklären, wenn man keine Elemente zum Einfügen hat. Wir lösen das ganz pragmatisch – in Kapitel 19.1 führen wir ein einfaches GUI-Element ein, einen Button. Danach besprechen wir die Layouts, und nutzen erstmal für alle Beispiele nur Buttons. Aber warum gibt es überhaupt Layouts? Man könnte doch einfach die GUI Elemente pixelgenau positionieren und fertig!? Dieses Vorgehen hat in der Praxis viele Probleme, da z.B. Schriftgrößen sehr unterschiedlich sind, und damit eine pixelgenaue Postionierung nicht sinnvoll ist. Außerdem können pixelgenaue Positionierungen z.B. sich nicht von alleine an wechselnde Fenster-Größen anpassen. Darum gibt es in Swing Layouts, die die Ausrichtung, Grösse und Position selbständig angleichen. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 349 / 409 19.1 Swing Klasse „JButton“ Für einen normalen Button gibt es in Swing die Klasse „JButton“ im Package „javax.swing“. Um einen Button mit Text auf der Schaltfäche zu erzeugten, muß der Konstruktor „JButton(String text)“ benutzt werden. Weitere Informationen zur Klasse „JButton“ finden sie in Kapitel 20.3. Ein GUI-Element wird in das Fenster mit „getContentPane().add“ eingefügt. import javax.swing.*; public class MyFrame extends JFrame { public MyFrame() { super("Fenster mit Button"); getContentPane().add(new JButton("Hallo")); } } Dieser Code erzeugt ein Fenster, dass komplett mit einem Button ausgefüllt ist. Und der Button passt sich automatisch der Größe des Fensters an. Probieren sie es aus! Abb. 19-1 : Ein Fenster mit einem Button 19.2 Grundlagen Ein „JFrame“ Fenster unterteilt sich in den sogenannten Client-Bereich und den Rest. Der Rest sind die Titelzeile, die Rahmen, möglicherweise Menü und Statusleiste, usw. Der innere Bereich steht dem Programm zur Verfügung und nennt sich Client-Bereich. Für die Verwaltung des Client-Bereichs enthält ein „JFrame“ Fenster ein sogenantes „ContentPane“ Objekt - vom Typ her ist es ein „java.awt.Container“. • Zugreifen kann man auf die Content-Pane mit der Funktion „getContentPane()“. • In diese Content-Pane können GUI-Elemente mit „add“ eingefügt werden. • Ohne Layout verwaltet die Content-Pane nur das zuletzt eingefügte Element. Im folgenden Programm z.B. enthält das Fenster nur den Button mit dem Text „3“. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 350 / 409 Abb. 19-2 : Die Content-Pane enthält nur den letzten Button import javax.swing.*; public class MyFrame extends JFrame { public MyFrame() { super("Nur der letzte Button \"3\" gewinnt"); getContentPane().add(new JButton("1")); getContentPane().add(new JButton("2")); getContentPane().add(new JButton("3")); } } Hinweis – GUI-Elemente im Sinne von Swing sind alle Objekte, die von „java.awt.Component“ abgeleitet sind. 19.3 Layouts Layouts sind Klassen, die mehrere GUI Elemente verwalten können, und diese aneinander positionieren. Folgende Layouts sind u.a. Teil der Swing Bibliothek: • Border-Layout • Flow-Layout • Grid-Layout Swing enthält noch viel mehr Layout-Klassen, aber diese drei sind mit Abstand am einfachsten zu benutzen und decken schon viele Anwendungs-Fälle ab. Aus Zeitmangel werden wir uns daher auf diese drei beschränken. 19.3.1 Border-Layout Das Border-Layout teilt sich in einen großen mittleren Bereich (CENTER) und vier SeitenBereiche (NORTH, SOUTH, WEST und EAST) auf. Beim Hinzufügen eines Elements muss der Ziel-Bereich angegeben werden – dafür existieren entsprechende Konstanten in der Klasse „BorderLayout“. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 351 / 409 Abb. 19-3 : Fenster mit Border-Layout import java.awt.*; import javax.swing.*; public class MyFrame extends JFrame { public MyFrame() { super("Fenster mit BorderLayout"); setSize(300, 200); Container c = getContentPane(); c.setLayout(new BorderLayout()); c.add(new JButton("North"), BorderLayout.NORTH); c.add(new JButton("West"), BorderLayout.WEST); c.add(new JButton("East"), BorderLayout.EAST); c.add(new JButton("South"), BorderLayout.SOUTH); c.add(new JButton("Center"), BorderLayout.CENTER); } } Beim Border-Layout müssen nicht alle Bereiche gesetzt werden. Sind welche unbesetzt, dann werden die anderen einfach vergrößert – bis der Client-Bereich abgedeckt ist. 19.3.2 Flow-Layout Das Flow-Layout ordnet Komponenten zeilenweise von links nach rechts und von oben nach unten an, wobei es preferredSize für jede Komponente verwendet. Es werden so viele Komponenten wie möglich in eine Zeile gesetzt, bevor eine neue Zeile begonnen wird. Abb. 19-4 : Fenster mit Flow-Layout import java.awt.*; import javax.swing.*; © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 352 / 409 public class MyFrame extends JFrame { public MyFrame() { super("Fenster mit FlowLayout"); setSize(320, 170); Container c = getContentPane(); c.setLayout(new FlowLayout()); c.add(new JButton("Schalter 1")); c.add(new JButton("Schalter 2")); c.add(new JButton("Schalter 3")); c.add(new JButton("Schalter 4")); c.add(new JButton("S 5")); c.add(new JButton("S 6")); c.add(new JButton("Ein sehr sehr sehr langer Schalter 7")); c.add(new JButton("Schalter 8")); } } 19.3.3 Grid-Layout Das Grid-Layout fügt Komponenten in ein Gitter von Zellen, bestehend aus Zeilen und Spalten, ein. Es vergrössert die Komponenten auf den in der Zelle verfügbaren Platz. Jede Zelle hat dieselbe Grösse. Das Gitter ist einheitlich. Wenn Sie die Grösse eines Grid-Layout-Containers verändern, vergrössert Grid-Layout die Zellen im Rahmen des für den Container verfügbaren Platzes auf das grösstmögliche Mass. Abb. 19-5 : Fenster mit Grid-Layout import java.awt.*; import javax.swing.*; public class MyFrame extends JFrame { public MyFrame() { super("Fenster mit GridLayout"); setSize(320, 170); Container c = getContentPane(); c.setLayout(new GridLayout(2, 3)); c.add(new JButton("1")); c.add(new JButton("2")); c.add(new JButton("3")); c.add(new JButton("4")); c.add(new JButton("5")); c.add(new JButton("6")); } } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 353 / 409 19.4 Verschachtelte Layouts Um Komponenten „tiefer zu layouten“, können Panels als Komponenten in ein Layout eingeführt werden. Panels können ihrerseits wieder ein Layout haben und Komponenten aufnehmen. Panels sind Objekte vom Typ „javax.swing.JPanel“ und auch ganz normale GUI Elemente – siehe auch Kapitel 20.8. Das folgende Beispiel zeigt ein Fenster mit Border-Layout, bei dem im Center-Bereich ein GridLayout integriert ist. Abb. 19-6 : Fenster mit verschachtelten Layouts import java.awt.*; import javax.swing.*; public class MyFrame extends JFrame { public MyFrame() { super("Fenster mit verschachtelten Layouts"); setSize(400, 300); JPanel panel = new JPanel(new GridLayout(3, 2)); panel.add(new JButton("1")); panel.add(new JButton("2")); panel.add(new JButton("3")); panel.add(new JButton("4")); panel.add(new JButton("5")); panel.add(new JButton("6")); Container c = getContentPane(); c.setLayout(new BorderLayout()); c.add(new JButton("North"), BorderLayout.NORTH); c.add(new JButton("West"), BorderLayout.WEST); c.add(new JButton("East"), BorderLayout.EAST); c.add(new JButton("South"), BorderLayout.SOUTH); c.add(panel, BorderLayout.CENTER); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 354 / 409 } } Hinweis – in Kapitel 16.6 finden Sie dieses Programm auch mit Windows-Style. 19.5 Aufgaben 19.5.1 Aufgabe „Layouts 1“ Entwickeln sie ein Fenster, dass oben zwei Reihen mit je 6 Buttons enthält, und den Rest des Bildschirms „frei“ läßt. Lösung siehe Kapitel 19.6. 19.5.2 Aufgabe „Layouts 2“ Entwickeln sie ein Fenster, dass an den Seiten je 5 Buttons unter einander enthält, und im restlichen Mittelbereich einen einzelnen Button. Lösung siehe Kapitel 19.7. 19.6 Lsg. zu Aufgabe „Layouts 1“ – Kap. 19.5.1 todo... 19.7 Lsg. zu Aufgabe „Layouts 2“ – Kap. 19.5.2 todo... 20 Swing GUI-Elemente In diesem Kapitel werden kurz einige Swing GUI-Elemente vorgestellt. In Swing gibt es natürlich noch viel mehr fertige GUI-Elemente – hier können nur ein paar kleine wichtige GUIElemente vorgestellt werden. Und vor allem können all diese GUI-Elemente viel mehr, als hier beschrieben wird. Die Kapitel stellen nur kurze Einführungen in die wichtigsten GrundFunktionalitäten der GUI-Elemente dar. 20.1 Labels Die Klasse „javax.swing.JLabel“ ist eine Klasse für einfache Labels (Beschriftungen). Wichtige Element-Funktionen: © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Konstruktor JLabel () Konstruktor JLabel (String text) void setText(String text) String getText() void setEnabled(boolean b) Seite 355 / 409 Erzeugt ein einfaches Label. Erzeugt ein Label mit Text. Setzt den Text des Labels neu. Gibt den Text des Labels zurück. Aktiviert bzw. deaktiviert das Label. Das Beispiel ist ein einfaches Fenster mit Label. Abb. 20-1 : Fenster mit Label import javax.swing.*; public class MyFrame extends JFrame { public MyFrame() { super("Fenster mit Label"); getContentPane().add(new JLabel("Ich bin ein Label")); } } 20.2 Text-Felder Die Klasse „javax.swing.JTextField“ ist eine Klasse für Text-Eingabe-Felder – auch Edit-Felder genannt. Wichtige Element-Funktionen: Konstruktor JTextField() Konstruktor JTextField (String text) void setText(String text) String getText() void setEnabled(boolean b) Erzeugt ein leeres Text-Feld. Erzeugt ein Text-Feld mit Text. Setzt den Text neu. Gibt den Text zurück. Aktiviert bzw. deaktiviert das Text-Feld. Wichtige Events: • Caret • Key Name: Beschreibung: Listener-Interface: Eine Funktion: Adapter-Klasse: Event-Klasse: Caret Wird aktiviert, wenn das Caret bewegt wird. javax.swing.event.CaretListener void caretUpdate(CaretEvent e) --javax.swing.event.CaretEvent Name: Key © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Beschreibung: Listener-Interface: Drei Funktionen: Adapter-Klasse: Event-Klasse: Seite 356 / 409 Wird aktiviert, wenn die Tastatur benutzt wird. java.awt.event.KeyListener void keyPressed(KeyEvent e) void keyReleased(KeyEvent e) void keyTyped(KeyEvent e) java.awt.event.KeyAdapter java.awt.event.KeyEvent Das Beispiel zeigt automatisch bei jeder Änderung des eingegebenen Textes im unteren Label die Anzahl von Zeichen des Textes an. Abb. 20-2 : Fenster mit Text-Feld und automatischer Längen-Angabe import import import import java.awt.*; java.awt.event.*; javax.swing.*; javax.swing.event.*; public class MyFrame extends JFrame { private JTextField field = new JTextField(); private JLabel output = new JLabel(); public MyFrame() { super("Fenster mit Text-Feld"); onTextChanged(); field.addCaretListener(new CaretListener() { public void caretUpdate(CaretEvent e) { onTextChanged(); } }); field.addKeyListener(new KeyAdapter() { public void keyTyped(KeyEvent e) { onTextChanged(); } }); Container c = getContentPane(); c.setLayout(new GridLayout(3, 1)); c.add(new JLabel("Bitte geben sie Text ein:")); c.add(field); c.add(output); } private void onTextChanged() { String text = field.getText(); output.setText("Der Text ist " + text.length() + " Zeichen lang."); } } Hinweis – es werden sowohl „Update-Caret“ als auch „Key-Typed“ Events überwacht, da je © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 357 / 409 nach Benutzung von „Entf“, „Backspace“, dem Clipboard, usw. unterschiedliche Events getriggert werden, und daher die Überwachung eines Events nicht ausreicht. 20.3 Buttons Die Klasse „javax.swing.JButton“ ist eine Klasse für normale Buttons, auch „Push-Buttons“ genannt. Wichtige Element-Funktionen: Konstruktor JButton() Konstruktor JButton(String text) void setText(String text) String getText() void setEnabled(boolean b) Erzeugt einen einfachen Button mit leerer Schaltfläche. Erzeugt einen Button mit Text auf der Schaltfläche. Setzt den Text der Schaltfäche neu. Gibt den Text der Schaltfäche zurück. Aktiviert bzw. deaktiviert den Button. Wichtige Events: • Action Name: Beschreibung: Listener-Interface: Eine Funktion: Adapter-Klasse: Event-Klasse: Action Wird aktiviert, wenn der Button betätigt wird. java.awt.ActionListener void actionPerformed(ActionEvent e) --java.awt.ActionEvent Das Beispiel ist ein Fenster mit einem Button „Klick mich“. Wird der Button angeklickt, so ändert sich der Text auf „Aua“, und bei jedem weiteren Klick wird ein weiteres „aua“ hinzugefügt. Abb. 20-3 : Fenster mit Button im initialen Zustand und nach dreimaliger Betätigung import java.awt.event.*; import javax.swing.*; public class MyFrame extends JFrame { private JButton button; public MyFrame() { © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 358 / 409 super("Fenster mit Button"); button = new JButton("Klick mich"); button.addActionListener(new ActionListener() { public void actionPerformed(ActionEvent e) { onButtonClick(); } }); getContentPane().add(button); } private void onButtonClick() { if (button.getText().equals("Klick mich")) { button.setText("Aua"); } else { button.setText(button.getText() + ", aua"); } } } 20.4 Radio-Buttons Die Klasse „javax.swing.JRadioButton“ ist eine Klasse für Radio-Buttons, d.h. Buttons die gruppiert auftreten, und von denen nur einer pro Gruppe selektiert sein kann. Default-mässig ist erstmal jeder Radio-Button seine eigene Gruppe, d.h. alle Radio-Buttons schalten unabhängig voneinander an und aus. Sollen mehrere Radio-Buttons einander automatisch deselektieren, so müssen sie einer gemeinsamen Button-Gruppe zugeordnet werden – hierfür gibt es in Swing die Klasse „javax.swing.ButtonGroup“. Mit „add“ können Radio-Button einer Gruppe hinzugefügt werden. Wichtige Element-Funktionen: Konstruktor JRadioButton() Konstruktor JRadioButton(String text) Ko. JRadioButton(String text, boolean b) void setText(String text) String getText() void setEnabled(boolean b) void setSelected(boolean b) boolean isSelected() Erzeugt einen unselektierten Radio-Button ohne Text. Erzeugt einen unselektierten Radio-Button mit Text. Erzeugt einen Radio-Button mit Text und entsprechender Selektion. Setzt den Text neu. Gibt den Text zurück. Aktiviert bzw. deaktiviert den Radio-Button. Selektiert bzw. deselektiert den Radio-Button. Gibt zurück, ob der Radio-Button selektiert ist. Wichtige Events: • Action – siehe Kapitel 20.3 Das Beispiel zeigt ein Fenster mit vier Radio-Buttons, die einer Gruppe zugeordnet sind. Ausserdem ist am unteren Rand ein Label vorhanden, das den jeweils selektieren Radio-Button explizit angibt – hierfür ist im Beispiel für jeden Radio-Button ein „ActionListener“ gesetzt. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 359 / 409 Abb. 20-4 : Fenster mit einer Gruppe von Radio-Buttons import java.awt.*; import java.awt.event.*; import javax.swing.*; public class MyFrame extends JFrame { private JRadioButton[] buttons = new JRadioButton[] { new JRadioButton("Eins", true), new JRadioButton("Zwei"), new JRadioButton("Drei"), new JRadioButton("Vier"), }; private ButtonGroup group = new ButtonGroup(); private JLabel output = new JLabel(); public MyFrame() { super("Fenster mit Radio-Buttons"); onSelectionChanged(buttons[0]); Container c = getContentPane(); c.setLayout(new GridLayout(buttons.length+1, 1)); for (int i=0; i<buttons.length; i++) { buttons[i].addActionListener(new ActionListener() { public void actionPerformed(ActionEvent e) { onSelectionChanged((JRadioButton)e.getSource()); } }); group.add(buttons[i]); c.add(buttons[i]); } c.add(output); } private void onSelectionChanged(JRadioButton button) { output.setText("Radio-But. \"" + button.getText() + "\" ist selektiert."); } } Achtung – diese Gruppierung via „ButtonGroup“ ist eine rein logische Gruppierung, und hat überhaupt nichts mit der Anordnung der Radio-Buttons im Fenster zu tun. 20.5 Check-Boxen Check-Boxes sind spezielle Buttons, die meistens zwei Stati haben – diese werden im GUI durch Boxen mit Check-Häkchen dargestellt. In Swing werden sie durch die Klasse „javax.swing.JCheckBox“ repräsentiert. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 360 / 409 Wichtige Element-Funktionen: Konstruktor JCheckBox() Konstruktor JCheckBox (String text) Kon. JCheckBox (String text, boolean b) void setText(String text) String getText() void setEnabled(boolean b) void setSelected(boolean b) boolean isSelected() Erzeugt eine unselektierte Check-Box ohne Text. Erzeugt eine unselektierte Check-Box mit Text. Erzeugt eine Check-Box mit Text und entsprechender Selektion. Setzt den Text neu. Gibt den Text zurück. Aktiviert bzw. deaktiviert die Check-Box. Selektiert bzw. deselektiert die Check-Box. Gibt zurück, ob die Check-Box selektiert ist. Wichtige Events: • Action – siehe Kapitel 20.3 Das Beispiel zeigt ein Fenster mit drei Check-Boxen. Werden sie selektiert bzw. deselektiert, so wird automatisch ihre Vordergrund-Farbe auf „grün“ bzw. „rot“ gesetzt. Abb. 20-5 : Fenster mit drei Check-Boxen import java.awt.*; import java.awt.event.*; import javax.swing.*; public class MyFrame extends JFrame { private JCheckBox[] buttons = new JCheckBox[] { new JCheckBox("Eins", true), new JCheckBox("Zwei"), new JCheckBox("Drei") }; public MyFrame() { super("Fenster mit Check-Boxen"); onSelectionChanged(buttons[0]); Container c = getContentPane(); c.setLayout(new GridLayout(buttons.length, 1)); for (int i=0; i<buttons.length; i++) { buttons[i].addActionListener(new ActionListener() { public void actionPerformed(ActionEvent e) { onSelectionChanged((JCheckBox)e.getSource()); } }); onSelectionChanged(buttons[i]); c.add(buttons[i]); } } private void onSelectionChanged(JCheckBox button) { button.setForeground(button.isSelected() ? Color.GREEN : Color.RED); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 361 / 409 } } 20.6 Scroll-Bars und Scroll-Panes Möglicherweise ist dem ein oder anderen aufgefallen, dass unsere Fenster noch ein kleines Problem haben – macht man sie zu klein, so verschwinden Elemente. Die Layouts haben Grenzen, sobald ein Element so klein würde, dass es nicht mehr sinnvoll darstellbar wäre. Möchte man in einem solchen Fall Scroll-Bars bekommen, so muss man sich nicht selber darum kümmern, sondern muss eine sogenannte Scroll-Pane zwischen Container und den enthaltenden Elementen legen. Im Extremfall ist dies sogar für ein einzelnes GUI-Element nötig, z.B. Tabellen – siehe Kapitel 20.7.2. Ein Scroll-Pane ist eine Fläche, die Scrollbars zur Verfügung stellt – in Swing dargestellt durch die Klasse „javax.swing.JScrollPane“ – ein Beispiel findet sich in Kapitel 20.7.2. Sowohl für die horizontale als auch für die vertikale Richtung können die Scrollbars auf verschiedene Arten unabhängig voneinander immer, nie, automatisch auf- und ausgeblendet werden – siehe auch das Interface „ScrollPaneConstants“. Bemerkung – hier sieht man wieder eine ganz zentrale OO-Philosophie, die sich natürlich auch in Swing wiederfindet: Baue keine „ich-kann-alles“ Klassen, sondern zerlege das Problem in viele kleine Komponenten, die man überschreiben und/oder wiederverwenden kann. 20.7 Tabellen Die Swing Tabellen-Klasse „javax.swing.JTable“ ist sehr leistungsfähig. Man kann quasi alles beeinflussen – bis hin zum Erscheinungs-Bild und den Editier-Möglichkeiten einer einzelnen Zelle. Für alle eigenständigen Aufgaben gibt es eigene Interfaces und Klassen, von denen man sich ableiten kann, und damit die Default-Einstellungen überschreiben kann. Leider sind die Tabellen dadurch nicht immer einfach in ihrer Programmierung – und ein wirklich umfassender Einstieg wäre fast ein eigenes Buch. Von daher beschränkt sich dieses Kapitel auf die wesentlichen Themen. 20.7.1 Eine einfache Tabelle Um eine einfache Tabelle zu erstellen, braucht es nicht viel. Man erzeugt ein Objekt der Klasse „javax.swing.JTable“ und übergibt dem Konstruktor die Anzahl an Zeilen und Spalten – im Beispiel 4 Zeilen und 3 Spalten. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 362 / 409 Abb. 20-6 : Fenster mit einfacher 4x3 Tabelle import javax.swing.*; public class MyFrame extends JFrame { public MyFrame() { super("Fenster mit Tabelle 1"); JTable table = new JTable(4, 3); setContentPane(table); } } 20.7.2 Tabellen mit Scrollbars Leider hat eine solche Tabelle ein Problem, das man erst bei einer größeren Anzahl an Zeilen bzw. Spalten sieht – darum hier eine Tabelle mit 40 Zeilen und 30 Spalten. Wie man sieht, hat die Tabelle keine Scrollbars. Abb. 20-7 : Fenster mit 40x30 Tabelle ohne Scrollbar import javax.swing.*; public class MyFrame extends JFrame { public MyFrame() { super("Fenster mit Tabelle 2"); JTable table = new JTable(40, 30); setContentPane(table); } } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 363 / 409 Wie wir in Kapitel 20.6 gelernt haben, müssen wir ein Scroll-Pane einsetzen. Wir machen dies hier in einer ganz primitiven Variante, ohne die Scrollbars hier wirklich optimal zu unterstützen. Abb. 20-8 : Fenster mit 40x30 Tabelle mit Scrollbar import javax.swing.*; public class MyFrame extends JFrame { public MyFrame() { super("Fenster mit Tabelle 3"); JTable table = new JTable(40, 30); JScrollPane scrollPane = new JScrollPane(table); setContentPane(scrollPane); } } 20.7.3 Tabellen-Inhalt aus Arrays Als wichtigstes fehlt aber noch der Inhalt der Tabellen – bislang sind sie leer und damit nicht besonders hilfreich. Um Inhalte in eine Tabelle anzuzeigen, muss ein Tabellen-Modell erstellt werden – siehe Kapitel 20.7.4. Da dies für einfache Tabellen-Anzeigen sehr aufwändig ist, gibt es zwei „Hilfs“ Konstruktoren in „JTable“, die Arrays bzw. den Java-Container „java.util.Vector“ erwarten. Im Hintergrund baut das „JTable“ aus dem Array oder demContainer ein Default-Tabellen-Modell auf – siehe „javax.swing.table.DefaultTableModel“. Im Beispiel bekommt „JTable“ 2 Arrays übergeben: • Parameter 1 ist ein zwei-dimensionales Array, das den Inhalt der Tabelle enthält – typisiert ist er auf „Object[ ][ ]“. Da sich jedes Objekt immer in einen String wandeln läßt, kann dieses Array immer angezeigt werden. • Parameter 2 ist ein ein-dimensionales Array, das die Spalten-Überschriften enthält. Auch es ist auf „Object[ ]“ typisiert. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 364 / 409 Abb. 20-9 : Fenster mit Tabelle, dessen Inhalt aus einem Array stammt import javax.swing.*; public class MyFrame extends JFrame { public MyFrame() { super("Fenster mit Tabelle 4"); String[][] values = {{"1","2"}, {"3","4"}, {"5","6"}, {"7","8"}}; String[] titles = { "Col 1", "Col 2"}; JTable table = new JTable(values, titles); setContentPane(new JScrollPane(table)); } } 20.7.4 Tabelle mit Tabellen-Modell Für komplexere Inhalte, Interaktion, und weitreichende Einflußmöglichkeiten muss der Tabelle ein Tabellen-Modell mitgegeben werden – d.h. das Objekt muss das Interface „javax.swing.table.TableModel“ implementieren. In diesem Interface ist die minimale Schnittstelle zwischen Tabellen-Anzeige und Tabellen-Daten beschrieben, so z.B. Funktionen für die Anzahl an Zeilen und Spalten, für die Inhalte, und einiges mehr. In der Praxis werden nicht alle diese Beeinflussungs-Möglichkeiten benötigt – d.h. gibt es die abstrakte Klasse „javax.swing.table.AbstractTableModel“, die für einige Funktionen DefaultImplementierungen anbietet. So müssen nur noch minimal drei Funktionen überschrieben werden. Das folgende Beispiel nutzt genau diese Klasse, leitet sich von „AbstractTableModel“ ab, und überschreibt den minimalen Satz an notwendigen Funktionen. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 365 / 409 Abb. 20-10 : Fenster mit Tabelle mit Tabellen-Modell import javax.swing.*; public class MyFrame extends JFrame { public MyFrame() { super("Fenster mit Tabelle 5"); JTable table = new JTable(new MyTableModel()); setContentPane(new JScrollPane(table)); } } import javax.swing.table.*; class MyTableModel extends AbstractTableModel { public int getRowCount() { return 15; } public int getColumnCount() { return 3; } public Object getValueAt(int arg0, int arg1) { return "" + arg0 + " / " + arg1; } } 20.8 Panels Auch die in Kapitel 19.4 vorgestellten Panels „javax.swing.JPanel“ sind ganz normale GuiElemente. Häufig werden sie einfach als eine Art GUI-Container in Verbindung mit Layouts genutzt. Aber sie können auch als einfache GUI-Elemente benutzt werden, die z.B. eigene Zeichnungen enthalten – natürlich via Ableiten und Überschreiben der „paint“ Funktion. 20.9 Menüs Um an ein „JFrame“ Fenster ein Menü anzuhängen, müssen drei Klassen benutzt werden: • „javax.swing.JMenuBar“ repräsentiert das komplette Menü, d.h. die Menü-Zeile im Fenster. • „javax.swing.JMenu“ repräsentiert einzelne Menüs, die Items, Seperatoren und Sub-Menüs enthalten können. Sub-Menüs sind wiederrum nur ganz normale Menüs – sie lassen sich halt verschachten. • „javax.swing.JMenuItem“ sind einzelne Menü-Einträge, die vom Benutzer angewählt werden können. Wichtige Element-Funktionen von „JMenuBar“ © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Konstruktor JMenuBar() void add(JMenu m) Seite 366 / 409 Erzeugt eine leere Menü-Zeile. Fügt das Menü ans Ende der Menü-Zeile an. Wichtige Element-Funktionen von „JMenu“ Konstruktor JMenu(String text) void add(JMenuItem item) void add(String text) void addSeparator() void setText(String text) String getText() void setEnabled(boolean b) Erzeugt ein leeres Menü mit dem übergebenen Text. Fügt den Menü-Eintrag ans Ende des Menüs an. Hier dürfen auch „JMenu“ Objekte übergeben werden, da diese von „JMenuItem“ abgeleitet sind. Fügt einen Menü-Eintrag mit dem übergebenen Text ans Ende des Menüs an. Kurzform für „add(new MenuItem(text))“. Fügt einen Separator ans Ende des Menüs an. Setzt den Text des Menüs neu. Gibt den Text des Menüs zurück. Aktiviert bzw. deaktiviert das Menü. Wichtige Element-Funktionen von „JMenuItem“ Konstruktor JMenuItem(String text) void setEnabled(boolean b) Erzeugt einen Menü-Eintrag mit dem übergebenen Text. Aktiviert bzw. deaktiviert den Menü-Eintrag Wichtige Events von „JMenuItem: • Action – siehe Kapitel 20.3 Um ein Menü ohne spezielle Features zu erstellen, muß man also: • eine Menü-Zeile erstellen, • mindestens ein Menü an die Menü-Zeile anhängen, • Menü-Einträge, Separatoren und Unter-Menüs an das Menü anhängen, und • die Menü-Einträge mit Action-Listener versehen. Hinweis – um Menü-Einträge selektiert, d.h. mit einem Häkchen versehen, darzustellen muss statt eines „JMenuItem“ ein Objekt der abgeiteten Klasse „JCheckBoxMenuItem“ genommen werden. Mit „void setSelected(boolean)“ kann die Selektion gesetzt, und mit „boolean isSelected()“ abgefragt werden. Das Beispiel ist ein „JFrame“ Fenster mit einer Menü-Zeile, die zwei Menüs enthält. Im Bild ist das erste Menü zu sehen – es besteht aus zwei Einträgen, einem Separator und einem SubMenü. Das Sub-Menü enthält drei Einträge, wobei der erste deaktiviert und der dritte selektiert ist. Das zweite Menü „Menü-2“ enthält nur einen Eintrag, an dem aber ein Action-Listener hängt. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 367 / 409 Abb. 20-11 : Fenster mit Menü import java.awt.event.*; import javax.swing.*; public class MyFrame extends JFrame { public MyFrame() { super("Fenster mit Menü"); JMenuItem subItem1 = new JMenuItem("Sub-Item 1"); JMenuItem subItem2 = new JMenuItem("Sub-Item 2"); JMenuItem subItem3 = new JCheckBoxMenuItem("Sub-Item 3"); subItem1.setEnabled(false); subItem3.setSelected(true); JMenu submenu = new JMenu("Sub-Menü"); submenu.add(subItem1); submenu.add(subItem2); submenu.add(subItem3); JMenu menu1 = new JMenu("Menü-1"); menu1.add("Item 1"); menu1.add("Item 2"); menu1.addSeparator(); menu1.add(submenu); JMenu menu2 = new JMenu("Menü-2"); JMenuItem item3 = new JMenuItem("Item 3"); item3.addActionListener(new ActionListener() { public void actionPerformed(ActionEvent e) { setTitle("Item 3 wurde betätigt"); } }); menu2.add(item3); JMenuBar menubar = new JMenuBar(); menubar.add(menu1); menubar.add(menu2); setJMenuBar(menubar); } } 20.10 Timer Für periodisch wiederkehrende Aufgaben gibt es in Swing eine Timer-Klasse „javax.swing.Timer“. Gestartet ruft sie in regelmäßigen Zeit-Intervallen – diese können in MilliSekunden definiert werden – die gesetzten Action-Listener auf. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 368 / 409 Hinweis –bei positiven Intervall-Werten muss der Timer explizit gestartet werden – siehe Tabelle und Beispiel. Wichtige Element-Funktionen: Konstr. Timer(int delay, ActionListener l) void start() void stop() Erzeugt einen Timer mit dem Zeit-Intervall „delay“ in MilliSekunden und dem übergebenen Action-Listener. Startet den Timer. Stoppt den Timer. Wichtige Events: • Action – siehe Kapitel 20.3 Das Beispiel erzeugt ein Fenster, dass jede halbe Sekunde die Farbe von „grün“ nach „rot“ und wieder zurück wechselt. Außerdem wird die aktuelle Farbe in der Titelzeile des Fensters angezeigt. Abb. 20-12 : Fenster in grün und in rot – gesteuert von einem Timer import java.awt.Color; import java.awt.event.*; import javax.swing.*; public class MyFrame extends JFrame { public MyFrame() { onTimer(); Timer timer = new Timer(500, new ActionListener() { public void actionPerformed(ActionEvent e) { onTimer(); } }); timer.start(); } private void onTimer() { if (getContentPane().getBackground() == Color.GREEN) { getContentPane().setBackground(Color.RED); setTitle("Fenster in \"rot\" mit Timer"); } else { getContentPane().setBackground(Color.GREEN); setTitle("Fenster in \"grün\" mit Timer"); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 369 / 409 } } } 20.11 Aufgaben 20.11.1 Aufgabe „Liste“ Programmieren sie ein Fenster mit einer Liste – Swing Klasse „JList“ – mit Scrollbars und Listen-Model. Die Benutzung der Listen-Klasse ist quasi analog zur Tabellen-Klasse, nur sind Liste und Listen-Modell einfacher, da eine Liste nur eine Spalte hat. Lösung siehe Kapitel 20.12. 20.11.2 Aufgabe „Scribble 6“ Implementieren sie zuerst ein Scribble, das sowohl Linienzüge als auch Rechtecke zeichnen kann. Setzen sie für die Auswahl Buttons oder ein Menü ein. Implementieren sie das Scribble in mehreren Schritten82: • Erste Version ohne Modell, mit dem Haupt-Augenmerk auf die Auswahl und die Integration von „Scribble 3“ – siehe Kapitel 18.3.5 – und „Scribble 5“ – siehe Kapitel 18.6.2. • In der zweiten Version müssen sie ein Modell entwickeln, das sowohl Linien-Züge als auch Rechtecke aufnehmen kann. Nehmen sie das Model vom „Sribble 4“ – siehe Kapitel 18.5 – erweitern es um Rechtecke, und integrieren es. Denken sie schon nach vorne – auf Dauer werden noch andere Grund-Elemente (Ellipsen, Texte,...) gezeichnet werden müssen. Am besten entwickeln sie also ein Modell, das offen ist für neue Grund-Elemente und mit verschiedenen Arten von Grund-Elementen umgehen kann. • Wenn sie eine gute offene Lösung für das Modell gefunden haben, ist die Integration einer weiteren Zeichen-Figur kein Problem. Also lassen sie ihr Scribble noch Ellipsen zeichnen können – natürlich zusätzlich zu den Polygonen und Rechtecken, und natürlich auch mit Model. An dem Aufwand, den sie treiben müssen, um Ellipsen zu integrieren können sie erkennen, wie gut ihr Programm-Design ist. Lösung siehe Kapitel 20.13. 20.11.3 Aufgabe „TextField“ Entwickeln sie ein eigenes Text-Feld Element, dass gegenüber dem Original-Element Ein guter Entwickler zeichnet sich nicht dadurch aus, dass er direkt am Computer die größten und tollsten Programme entwickelt, sondern dadurch, dass er gute Angewohnheiten hat, die ihn gut machen. Dazu gehört eben auch ein Problem in mehrere Stufen zu zerlegen und diese nach und nach abzuarbeiten. Aber auch so Dinge wie: Coding-Styles, Unit-Tests, Assertions, Refactoring, Weiterbildung, Reflektion, und einiges mehr. 82 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 370 / 409 „JTextField“ noch zusätzlich ein externes Event für die Änderung des Inhalts hat, d.h. einen Change-Listener. Lösung siehe Kapitel 20.14. 20.11.4 Aufgabe „Kontaktdaten 5“ Implementieren sie ein GUI für die „Kontaktdaten-Verwaltung 4“ aus Kapitel todo bzw. Kapitel. Lösung siehe Kapitel 20.15. 20.12 Lsg. zu Aufgabe „Liste“ – Kap. 20.11.1 todo... 20.13 Lsg. zu Aufgabe „Scribble 6“ – Kap. 20.11.2 todo... 20.14 Lsg. zu Aufgabe „TextField“ – Kap. 20.11.3 todo... 20.15 Lsg. zu Aufgabe „Kontaktdaten 5“ – Kap. 20.11.4 todo... 21 Applets Applets sind kleine Programme, die in eine HTML Seite eingebettet sind und von einem Browser ausgeführt werden. Achtung – gleich vorweg ein paar Hinweise zur praktischen Seite der Entwicklung von Applets: • Der verwendete Browser muss natürlich java-fähig sein. • Es muss ein entsprechendes Java-Plugin (passendes JDK) für den Browser installiert sein. • Es gibt immer wieder Ärger mit den Browser-Caches. D.h. die Browser merken nicht dass sie eine neue Class-Datei erzeugt haben, und benutzen die aus dem Cache. 21.1 Beispiel Sie benötigen die einbettende HTML Seite und die „class“ Datei des Applets im gleichen © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 371 / 409 Verzeichnis (wenn die Applet Klasse nicht in einem Package liegt). <html> <head> <meta HTTP-EQUIV="Content-Type" CONTENT="text/html; charset=iso-8859-1"> <title> Mein erstes Applet </title> </head> <body> <applet codebase = "." code = "MyApplet.class" weidth = 400 hight = 300 > </applet> </body> </html> import java.awt.*; import java.awt.event.*; import javax.swing.*; public class MyApplet extends JApplet { public MyApplet() { final JButton button = new JButton("Klick mich bitte vorsichtig an"); button.addActionListener(new ActionListener() { public void actionPerformed(ActionEvent e) { button.setText("Aua, das tat weh..."); } }); setContentPane(button); } } 21.2 Applet HTML-Seite Ein Applet kann nie für sich alleine funktionieren, sondern braucht immer eine HTML Seite, in die es eingebettet ist. Dies soll und kann kein HTML Lehrgang sein, darum wird eine kleine Beispiel Seite kurz besprochen - für mehr Details lesen sie bitte ein entsprechendes HTML Buch. <html> <head> <meta HTTP-EQUIV="Content-Type" CONTENT="text/html; charset=iso-8859-1"> <title> HTML-Testseite </title> </head> <body> Mein tolles Applet erscheint in einem Java-fähigen Browser.<BR> <applet CODEBASE = "." CODE = "packageName.MeineAppletKlasse.class" NAME = "TestApplet" WIDTH = 400 HEIGHT = 300 HSPACE = 0 VSPACE = 0 ALIGN = middle > © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 372 / 409 Text für den Fall, dass der Browser keine Applets anzeigen kannn </applet> </body> </html> Hinweis – aufgrund der Plattform-unabhängigkeit von HTML sollten alle „nicht-normalen“ Zeichen codiert sein, d.h. das „ä“ statt des einfachen „ä“. Auch diese Dinge sollten sie genauer in einem HTML Buch nachlesen. Für Java Applets wurde in HTML das Tag „applet“ aufgenommen, das mit <applet> begonnen und mit </applet> beendet wird. Der zwischen den Tags stehende Text wird vom Browser ausgegeben, wenn er nicht Applet-fähig ist. Zum Applet-Start-Tag gehören Parameter, die das Applet näher spezifizieren: Parameter optional Bedeutung CODE Gibt den Namen des Applets an: • Gross-/Kleinschreibung muss beachtet werden. • die Extension .class sollte angegeben werden. • die Klassendatei muss im aktuellen Verzeichnis liegen WIDTH Die für das Applet zur Verfügung stehende Breite HEIGHT Die für das Applet zur Verfügung stehende Höhe CODEBASE x Alternative Verzeichnisse (durch Kommata getrennt) für die Klassendateien ARCHIVE x Angabe eines jar-Archivs für die Klassendateien und die Ressourcen OBJECT x Datei mit dem serialisierten Inhalt des Applets ALT x Alternativer Text für nicht Appletfähige Browser NAME x Eindeutiger Name für das Applet. Wichtig bei der Verwendung mehrerer miteinander kommunizierender Applets auf einer Seite ALIGN x Vertikale Anordnung des Applets - einer der folgenden Werte (left, right, top, texttop, middle, absmiddle, baseline, bottom, absbottom) VSPACE x Rand über und unter dem Applet HSPACE x Rand rechts und links vom Applet © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 373 / 409 21.3 Grundlagen 21.3.1 Klassen-Hierarchie „Applet“ bzw. „JApplet“ Ein Applet muss zwingend von „java.applet.Applet“ abgeleitet sein. Wird Swing benutzt Ableitungs-Hierarchie: Object - Component - Container - Panel - Applet - JApplet 21.3.2 Default-Konstruktor Ein Browser erzeugt ein Applet immer mit dem Default-Konstruktor. In diesem sollten sie bei Applets aber ausnahmsweise keine Initialisierung vornehmen. Für die Initialisierung steht in Applets die Funktion „init“ zur Verfügung – siehe Kapitel 21.3.3.1. 21.3.3 Wichtige Applet-Funktionen Es gibt verschiedene Applet-Funktionen, die der Kommunikation zwischen Browser und Applet dienen und in einer eigenen Applet-Klasse überschrieben werden können. 21.3.3.1 Funktion „public void init()“ Nach der Erstellung des Applets wird für das Applet die Funktion „init“ vom Browser aufgerufen, in der Sie das Applet initailisieren sollten. Die Funktion wird genau einmal für das Applet nach der Erzeugung aufgerufen. 21.3.3.2 Funktion „public void start()“ Um die Ausführung des Applets zu starten ruft der Browser die Funktion „start“ auf. Achtung - diese Funktion kann mehrfach aufgerufen werden. Z. B. bei einem Seitenwechsel muss der Browser das Applet nicht zerstören, sondern kann es cachen. Wird die Seite wieder aufgerufen, muss der Browser das Applet daher nicht neu erzeugen und ruft dann natürlich auch nicht mehr die Funktion „init“ auf, sondern direkt wieder „start“. 21.3.3.3 Funktion „public void stop()“ Die Funktion „stop“ ruft der Browser auf, wenn das Applet gestoppt werden soll. Achtung - wie schon start, so kann auch stop mehrfach aufgerufen werden, da der Browser nur die Ausführung des Applets unterbricht, aber das Applet nicht zerstört, z. B. beim Verlassen der HTML Seite mit cachen des Applets. Geben sie also niemals Ressourcen, die beim Starten wieder benötigt werden, in dieser Funktion frei. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 374 / 409 21.3.3.4 funktion „public void destroy()“ Erst die Funktion „destroy“ zeigt an, dass das Applet zerstört wird. Auch diese Funktion wird nur einmal (wie init) am Ende der Lebensdauer des Applets vom Browser aufgerufen. Achtung - für die Freigabe von Ressourcen benutzen sie immer destroy und nicht den Destruktor finalize, da dieser nicht zwingend aufgerufen wird. 21.3.3.5 Funktion „public void showStatus(String msg)“ Mit der funktion „showStatus“ können sie Ausgaben in der Statuszeile des Browsers vornehmen. showStatus("Hallo Welt in der Statuszeile"); 21.3.3.6 Funktion „public String getAppletInfo()“ Mit dieser funktion können sie Informationen über Ihr Applet zur Verfügung stellen, die der Browser abrufen und anzeigen kann. public String getAppletInfo() { return "Mein Super-Applet Version 0.001"; } 21.4 Applet-Parameter Neben den Parametern für das Applet-Tag selber können dem Applet auch noch Appletspezifische Parameter übergeben werden. Diese werden mit dem optionalen HTML-Tag PARAM im Applet-Bereich angegeben - PARAM wiederum hat die Parameter name und value. <APPLET CODEBASE CODE NAME WIDTH HEIGHT HSPACE VSPACE ALIGN > <PARAM <PARAM </APPLET> = = = = = = = = "." "packageName.meineAppletKlasse.class" "TestApplet" 400 300 0 0 middle NAME = "speed" VALUE = "10"> NAME = "rate" VALUE = "2"> Damit das Applet die Parameter aus der HTML Seite erfragen kann, gibt es die Funktion public String getParameter(String name), die den Namen des Parameters enthält und den Wert des Parameters als String zurückgibt. Für nähere Informationen über die Parameter kann das Applet die Funktion public String[ ][ ] getParameterInfo() überschreiben, die ein zweidimensionales String-Array zurückgibt - eine Zeile pro Parameter, den das Applet erwartet. Eine Zeile besteht aus drei Einträgen, die den Namen des Parameters, den Typ und eine Beschreibung enthalten. Die Einträge sollten für Menschen lesbar und verständlich sein, da der Browser diese Informationen nicht interpretiert, © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 375 / 409 sondern sie nur dem Benutzer zugänglich macht. public String[][] getParameterInfo() { String pinfo[][] = { {"speed", "int", "Schnelligkeit"}, {"rate", "int", "Rate"}, }; } 21.5 Applets & Applikationen Von einer „stand-alone“ Applikation unterscheidet sich ein Applet durch mehrere Dinge: Applet „stand-alone“ Applikation Es muss eine spezielle Applet-Klasse geben, Es wird keine spezielle Applikations-Klasse die direkt oder indirekt83 von der Klasse zwingend vorausgesetzt. Ein „main“ kann in „java.applet.Applet“ abgeleitet sein muss. jeder Klasse stehen. Wenn eine spezielle Applikations-Klasse vorhanden ist, so kann sie von einer beliebigen Klasse abgeleitet sein. Start, indem der Browser die Haupt-Klasse Start in der eindeutigen static Funktion main. des Applets instanziiert (Default-Konstruktor) und die Funktion init und start aufruft. Darf nicht auf Dateien auf dem lokalen Rechner zugreifen und darf keine Prozesse auf diesem starten84. Keine Sicherheitsbeschränkungen bzgl. des lokalen Systems Arbeitet immer grafik- und ereignissorientiert85. Kann auch Texteingabe und -ausgabe auf der Kommandozeile vornehmen. Hinweis – es ist relativ leicht Java-Sourcen zu schreiben, die sowohl als Applet als auch als „stand-alone“ Anwendung laufen. 83 Wenn Sie ein Applet basierend auf den Swing Klassen erstellen, so benutzen Sie als Basisklasse „javax.swing.JApplet“, die wiederum von „java.applet.Applet“ abgeleitet ist. 84 Genau genommen gibt es ein sehr detailiertes Sicherheits-Konzept, über das der Benutzer einem Applet beliebige Rechte geben kann. Defaultmäßig hat ein Applet aber keine Rechte auf dem Rechner des Benutzers. 85 In der Entwicklungsumgebung kann die Kommandozeile problemlos zum Debuggen benutzt werden. Aber in manchen Browsern existiert keine Kommandozeile, so dass Textein- und ausgabe keinen Sinn machen - viele Browser stellen ein Ausgabefenster für Textausgaben zur Verfügung, dass der Benutzer aber extra öffnen muss. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 376 / 409 21.6 Aufgaben 21.6.1 Aufgabe „Statuszeilen-Applet“ Implementieren sie ein Applet, bestehend aus einem Text-Eingabe-Feld und einem Button. Bei Betätigung des Buttons soll der Inhalt des Text-Feldes in der Statuszeile des Browsers ausgegeben werden. Lösung siehe Kapitel 21.7. 21.6.2 Aufgabe „Scribble 7“ Modifizieren sie das „Scribble 6“ – siehe Kapitel 20.11.2 – so, dass es auch als Applet läuft. Lösung siehe Kapitel 21.8. 21.7 Lsg. zu Aufgabe „Statuszeilen-Applet“ – Kap. 21.6.1 todo... 21.8 Lsg. zu Aufgabe „Scribble 7“ – Kap. 21.6.2 todo... 22 Exceptions Exceptions sind ein Sprachkonzept zur Behandlung von unnormalen Programm-Status, z.B. von Fehlern. 22.1 Motivation Die beiden grössten Probleme der konventionellen Fehlerbehandlung (Rückgabe von ErrorCode’s, Fehlerparameter, globale Variablen, Fehlerfunktionen,...) sind, dass sich im Code normaler Code und Fehlerbehandlungscode logisch vermischen, und die Abfrage auf Fehler von der Sorgfalt des Programmierers abhängig ist. Das Ergebnis sind Quelltexte folgender Art – Achtung, Pseudocode: back = fkt1(); if (back==ERROR) { // behandle Fehler } back = fkt2(); if (back==ERROR) { // behandle Fehler } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 fkt3(); int i = fkt4(); if (i<0) { back = fkt5().fkt6(); if (back==ERROR) { // behandle Fehler } else { back = fkt7(); if (back==ERROR) { // behandle Fehler } } } Seite 377 / 409 // Fehlerbehandlung vergessen // Behandlung vergessen oder keine Fehler moeglich? // Behandlung vergessen oder keine Fehler moeglich? Schöner wäre aber: // Normaler Code fkt1(); fkt2(); fkt3(); int i = fkt4(); if (i<0) { fkt5().fkt6(); } else { fkt7(); } // Fehlerbehandlung if (ERROR) { // behandle Fehler } Aus diesen Gründen heraus entstand das Konzept der Exceptions. 22.2 Realisation Tritt ein Fehler auf, so wird ein Exception-Objekt geworfen – dies darf nur „logisch innerhalb86“ eines try-Blocks passieren. Wird ein Exception-Objekt geworfen, wird instantan in die entsprechende Fehler-Behandlung verzweigt. public static void main(String[] args) { for (int i=0; i<2; i++) { try { System.out.println("try-Block Anfang mit i=" + i); if (i==1) { throw new RuntimeException(); } System.out.println("try-Block Ende"); } catch (RuntimeException x) { System.out.println("Fehler-Objekt wurde gefangen"); } } } (*) (**) Ausgabe try-Block Anfang mit i=0 86 „logisch innerhalb“ meint hier aus Sicht des Programm-Verlaufs – siehe auch Kapitel 22.3. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 378 / 409 try-Block Ende try-Block Anfang mit i=1 Fehler-Objekt wurde gefangen An der Ausgabe sieht man sehr schön, dass bei „i==0“ die Schleife normal durchlaufen un der catch-Block ignoriert wird. Bei „i==1“ dagegen wird direkt nach dem „throw“ die Abarbeitung des try-Blocks unterbrochen und in den catch-Block verzweigt. Man kann also sagen: • Der try-Block ist der Bereich, der den normalen Code ohne Fehlerbehandlung enthält. • Tritt im normalen Programm-Verlauf ein Fehler auf, so meldet die entsprechende Stelle diesen indem sie mit „throw“ ein Fehler-Objekt wirft – im Beispiel vom Typ „RuntimeException“ in Zeile (*). • Im Falle eines Fehlers, d.h. eines „throw’s“, wird der normale Programm-Verlauf automatisch unterbrochen und sofort zur entsprechenden Fehler-Behandlung verzweigt – im Beispiel der catch-Block (**). • Ein catch-Block stellt den Fehler-Behandlungs-Code zur Verfügung. • Nach Verlassen des catch-Blocks gilt der Fehler als behandelt. Die Idee hierbei ist, dass es nach Auftreten eines Fehler einfach keinen Sinn mehr macht den normalen Programm-Verlauf zu verfolgen. Würde der Fehler ignoriert und der normale Programm-Fluß weiter abgearbeitet werden, so kann dies nicht gut sein sondern statt dessen gefährliche Folgen haben. Denken sie z.B. ganz extrem, dass die Kühlung eines Kraftwerks nicht aktiviert werden konnte. Wenn dieser Fehler ignoriert werden würde, und die Generatoren danach einfach weiter hoch gefahren werden würde, so wäre das sicher nicht das Beste. Hinweis – zu einem try-Block gehört immer mindestens ein catch-Block oder ein ein finallyBlock – siehe auch Kapitel 22.4. Hinweis – in der Praxis arbeitet man eher selten mit der Klasse „RuntimeException“ als Exception-Klasse – siehe Zeile (*) und (**). Im Hinblick auf die nächsten Grundlagen ist die Klasse „RuntimeException“ aber erstmal einfacher, da sie uns Exception-Spezifikationen (siehe Kapitel todo) erspart. Welche Klassen für das Exception-Handling zur Verfügung stehen, und welche man wann benutzt, wird in Kapitel todo erklärt. 22.3 Fehler über mehrere Funktions-Ebenen Das Exception-Handling ist nicht auf eine Funktion festgelegt, sondern kann über eine beliebige Menge von Funktions-Aufrufen hinweg erfolgen. Das folgende Beispiel ist im Prinzip das Gleiche wie das Vorherige. Nur steht das „if“ mit dem „throw“ nicht direkt im try-Block, sondern wird über 2 Funktions-Aufrufe („f1“ und „f“) angeprochen. Die Funktionen enthalten zusätzlich noch Ausgaben, um den Programm-Verlauf gut verfolgen zu können. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 379 / 409 private static void f2(int i) { System.out.println(" -> f2"); if (i==1) { throw new RuntimeException(); } System.out.println(" <- f2"); } private static void f1(int i) { System.out.println(" -> f1"); f2(i); System.out.println(" <- f1"); } public static void main(String[] args) { for (int i=0; i<2; i++) { try { System.out.println("try-Block Anfang mit i=" + i); f1(i); System.out.println("try-Block Ende"); } catch (RuntimeException x) { System.out.println("Fehler-Objekt wurde gefangen"); } } } Ausgabe try-Block Anfang mit i=0 -> f1 -> f2 <- f2 <- f1 try-Block Ende try-Block Anfang mit i=1 -> f1 -> f2 Fehler-Objekt wurde gefangen Wie man an den Ausgaben deutlich sieht, arbeitet das Exception-Handling auch über mehrere Funktions-Ebenen hinweg. • Das throw steht weiterhin logisch in einem try-Block. Innerhalb der Funktion „f2“ ist zwar kein try-Block vorhanden, aber im logischen Sinne steht „f2“ in „f1“ und „f1“ im try-Block von „main“. • Nach werfen des Exception-Objekts wird auch hier automatisch und sofort in den FehlerBehandlungs Block verzweigt. Man sagt, dass eine Exception automatisch den FunktionsStack abbaut. Es wird kein Code mehr im normalen Programm-Verlauf ausgeführt – auf keiner Ebene der Funktionen, bis der Fehler als bearbeitet gilt. 22.4 Aufräumarbeiten und „finally“ Müssen nach einem Code-Abschnitt auf jeden Fall irgendwelche Aufräumarbeiten ausgeführt werden – z.B. das Schließen von Dateien oder Netzwerkverbindungen – so gibt es dafür einen finally-Block. Ein finally-Block gehört immer zu einem try-Block. Egal wie der try-Block verlassen wird (normaler Programmfluß, return im try-Block, Exception), der Code im finally Block wird immer ausgeführt. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 380 / 409 Das folgende Beispiel ist bis auf zwei Änderungen das Gleiche wie das Vorherige: • In der Funktion „f1“ ist der Aufruf von „f2“ in einen try-Block gelegt, und der zugehörige finally-Block enthält die Ausgabe zum Verlassen der Funktion. • Der try-Block in „main“ hat einen finally-Block mit einer zusätzlichen Ausgabe bekommen. private static void f2(int i) { System.out.println(" -> f2"); if (i==1) { throw new RuntimeException(); } System.out.println(" <- f2"); } private static void f1(int i) { System.out.println(" -> f1"); try { f2(i); } finally { System.out.println(" <- f1"); } } public static void main(String[] args) { for (int i=0; i<2; i++) { try { System.out.println("try-Block Anfang mit i=" + i); f1(i); System.out.println("try-Block Ende"); } catch (RuntimeException x) { System.out.println("Fehler-Objekt wurde gefangen"); } finally { System.out.println("finally-Block in main"); } } } Ausgabe try-Block Anfang mit i=0 -> f1 -> f2 <- f2 <- f1 try-Block Ende finally-Block in main try-Block Anfang mit i=1 -> f1 -> f2 <- f1 Fehler-Objekt wurde gefangen finally-Block in main An den Ausgaben kann man gut verfolgen, dass der finally-Block immer ausgeführt wird, wenn der zugehörige try-Block verlassen wird. Hinweise zu Syntax: • Zu einem try-Block gehört mindestens ein catch-Block oder ein finally-Block. • D.h. ist auch ein try-Block nur mit finally-Block erlaubt – siehe z.B. Funktion „f1“ im Beispiel. • Gehören zu einem try-Block catch-Blöcke und ein finally-Block, so folgt der finally-Block dem © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 381 / 409 letzten catch-Block, d.h. er steht am Ende des kompletten try-catch-finally-Konstrukts. Siehe auch „main“ im Beispiel. • Es darf nur einen finally-Block pro try-Block geben. 22.5 Exception-Objekte und Exception-Hierarchie 22.5.1 Exception-Objekte Es können nur Objekte der Klasse „java.lang.Throwable„ oder einer von ihr abgeleiteten Klasse Exception-Objekte sein, d.h. in einer throw-Anweisung geworfen werden. throw new String("Error"); throw new Exception(); // Fehler, String ist kein Throwable // Okay, Exception ist ein Throwable Der Sinn einer konkreten Exception Klasse ist, dass sie einen ganz fest umrissenden Fehler spezifiziert. So gibt es z.B. in Java die „java.lang.NullPointerException“, die geworfen wird, wenn auf eine Null-Referenz zugegriffen wird. Oder z.B. beim Zugriff auf nicht existente ArrayElemente wird eine „java.lang.ArrayIndexOutOfBoundsException“ geworfen. String s = null; s.toString(); // erzeugt eine NullPointerException int[] a = { 0, 1, 2 }; a[5] = 5; // erzeugt eine ArrayIndexOutOfBoundsException 22.5.2 Exception-Hierarchie In der Klassen-Bibliothek von Java existieren viele Exception-Klassen, d.h. Klassen die direkt oder indirekt von „Throwable“ abgeleitet sind. Von besonderer Bedeutung sind dabei die folgenden: java.lang.Throwable java.lang.Error java.lang.Exception java.lang. RuntimeException Abb. 22-1 : Ein kleiner aber wichtiger Teil der Exception-Hierarchie in Java Die Klasse „java.lang.Error“ und ihre Unterklassen sind für interne Probleme der virtuellen Maschine (z. B. beim Linken, Speicherüberlauf,...) reserviert. Sie müssen diese Exceptions nicht behandeln und sollten es auch nicht, da sie im Normalfall für den Programmierer nicht behandelbar sind. Sie sollten auch keine Unterklassen von „java.lang.Error“ bilden, da dies im Normallfall keinen Sinn macht. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 382 / 409 Die Klasse „java.lang.Exception“ ist die direkte oder indirekte Baisklasse für die normalen Exceptions. Auch sie sollten diese Klasse im Normallfall als direkte oder indirekte Baisklasse für ihre eigenen Exception-Klassen nutzen. Die Klasse „java.lang.RuntimeException“ ist besonders, da sie und ihre abgeleiteten Klassen ungeprüfte Exceptions ermöglichen – siehe Kapitel 22.9. 22.6 Fangen von Basis-Exception-Klassen Nicht immer will man aber ganz genau wissen, was für ein Fehler aufgetreten ist. Dann ist es möglich ganze Gruppen von Exceptions über ihre gemeinsame Basis-Klasse abzuhandeln. Eine Exception-Basis-Klasse faßt daher eine Menge von thematisch ähnlichen oder zusammen gehörigen Fehlern zusammen. Bei einer Datei-Schittstelle könnte es z.B. Exceptions der folgenden Art geben: • FileDoesNotExistException • FileNotReadableException • FileNotWritableException • ... Hier würde es sich anbieten, die Exception-Klassen von einer gemeinsamen Basis-Klasse „FileException“ abzuleiten. Abb. 22-2 : Klassen-Hierarchie für unsere Beispiel File-Exceptions Will man nun gar nicht genau wissen, welche File-Exception geworfen wurde, sondern nur dass überhaupt ein Datei-Fehler aufgetreten ist, so kann man im catch-Block einfach die BasisKlasse abfangen. public class FileException extends RuntimeException { } public class FileDoesNotExistException extends FileException { } public class FileNotReadableException extends FileException { } public class FileNotWritableException extends FileException { } public static void main(String[] args) { for (int i=0; i<4; i++) { try { System.out.println("try-Block mit i=" + i); if (i==0) { throw new FileDoesNotExistException(); } else if (i==1) { throw new FileNotReadableException(); } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 383 / 409 else if (i==2) { throw new FileNotWritableException(); } System.out.println("try-Block Ende"); } catch (FileException x) { System.out.println("FileException wurde gefangen"); } } } Ausgabe try-Block mit i=0 FileException wurde gefangen try-Block mit i=1 FileException wurde gefangen try-Block mit i=2 FileException wurde gefangen try-Block mit i=3 try-Block Ende Wie man sieht, wird in bei allen geworfenen Exception-Objekten der catch-Block der BasisKlasse zur Fehler-Behandlung genutzt. 22.7 Mehrere catch-Blöcke Was ist aber nun, wenn ich ein paar konkrete Fehler abfangen und bearbeiten will, andere aber ganz allgemein behandeln möchte? In den vorherigen Kapiteln wurde schon zwischen den Zeilen erwähnt, dass es mehrere catchBlöcke pro try-Block geben kann. • Jeder catch-Block ist für eine Exception-Klasse zuständig. • Wird ein Exception-Objekt geworfen, so werden die catch-Blöcke von oben nach unten abgesucht, ob das Exception-Objekt zu dem Typ des catch-Parameters paßt. Wenn ja, so wird in den catch-Block verzweigt. • Daher darf auch kein allgemeiner catch-Block vor einem spezielleren stehen. Wäre z.B. der catch-Block für „FileException“ der erste, und erst danach würde der für „FileDoesNotExistException“ folgen, so würde der zweite catch-Block nie zum Zuge kommen. Denn jede „FileDoesNotExistException“ würde schon vom „FileException“ catchBlock abgefangen werden. • Nach normaler Abarbeitung eines catch-Blocks gilt der Fehler als behandelt, und es wird – möglicherweise nach Bearbeitung des optionalen try-Blocks – nach dem letzten catch-Block fortgefahren. public static void main(String[] args) { for (int i=0; i<4; i++) { try { System.out.println("try-Block mit i=" + i); if (i==0) { throw new FileDoesNotExistException(); } else if (i==1) { throw new FileNotReadableException(); } else if (i==2) { © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 384 / 409 throw new FileNotWritableException(); } System.out.println("try-Block Ende"); } catch (FileDoesNotExistException x) { System.out.println("FileDoesNotExistException wurde gefangen"); } catch (FileNotReadableException x) { System.out.println("FileNotReadableException wurde gefangen"); } catch (FileException x) { System.out.println("FileException wurde gefangen"); } } } Ausgabe try-Block mit i=0 FileDoesNotExistException wurde gefangen try-Block mit i=1 FileNotReadableException wurde gefangen try-Block mit i=2 FileException wurde gefangen try-Block mit i=3 try-Block Ende 22.8 Unbehandelte Exceptions Wird ein Exception-Objekt geworfen, für das kein passender catch-Block, so wird der Funktions-Stack Funktion für Funktion abgebaut (siehe Kapitel 22.3) bis zur „main“ Funktion. Ist auch hier kein passender catch-Block vorhanden, so wird das Programm automatisch beendet. private static void f2() { System.out.println(" - f2"); System.out.println(" => throw ohne catch"); throw new RuntimeException(); } private static void f1() { System.out.println("- f1"); f2(); } public static void main(String[] args) { System.out.println("main"); f1(); } mögliche Ausgabe main - f1 - f2 => throw ohne catch java.lang.RuntimeException at Appl.f2(Appl.java:6) at Appl.f1(Appl.java:11) at Appl.main(Appl.java:16) Exception in thread "main" Durch dieses Vorgehen wird verhindert, dass ein Fehler-Zustand unbehandelt bleibt, aber das Programm weiter läuft – siehe auch Kapitel todo. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 385 / 409 22.9 Geprüfte und ungeprüfte Exceptions In Java wird zwischen geprüften und ungeprüften Exceptions unterschieden – sie heißen auch checked bzw. unchecked exceptions. Für geprüfte Exceptions kann der Compiler zur Compile-Zeit feststellen, ob sie vollständig behandelt werden, d.h. es immer einen passenden catch-Block im Quelltext gibt. Für ungeprüfte Exceptions gilt dies nicht. • Ungeprüfte Exceptions sind Exceptions, bei denen Objekte geworfen werden, die direkt oder indirekt von „java.lang.Error“ bzw. „java.lang.RuntimeException“ abgeleitet sind. • Alle anderen Exception-Objekte sind geprüfte Exceptions. Welche Konsequenzen hat aber nun die Verwendung von geprüften Exceptions? Der Code ist nur dann korrekt, d.h. wird vom Compiler compiliert, wenn das Werfen einer geprüften Exception in der Funktions selber abgefangen wird, oder mit einer Exception-Spezifikation nach außen gemeldet wird. public void f() { throw new Exception(); } // Compiler-Fehler, unbehandelte Exception Lösung 1 – Abfangen der Exception in der Funktion selber public void f() { try { throw new Exception(); } catch (Exception x) { } } // Exception wird abgefangen Lösung 2 – Funktion mit Exception-Spezifikation public void f() throws Exception { throw new Exception(); // Exception wird nach aussen gemeldet } Ein Exception-Spezifikation wird hinter die Paramter-Liste der Funktion geschrieben, und besteht aus dem Schlüsselwort „throws“ und einer Auflistung der möglichen Exceptions. Wird in der Exception-Spezifikation eine Klasse angegeben, so werden damit alle Objekte vom Typ der Basis-Klasse bzw. einer von ihr abgeleiteten Klassen als mögliche Exception-Objekte erlaubt. Ist keine Exception-Spezifikation vorhanden, so darf keine geprüfte Exception geworfen werden. Beispiele: - void f() throws Exception - void g() throws FileDoesNotExistException, FileNotReadableException - void h() • void f() throws Exception Diese Funktion darf alle Exceptions werfen, deren Typ „java.lang.Exception“ ist bzw. deren Typen direkt oder indirekt von „java.lang.Exception“ abgeleitet sind. Und natürlich ungeprüfte Exceptions. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 386 / 409 • void g() throws FileDoesNotExistException, FileNotReadableException Diese Funktion darf alle Exceptions werfen, deren Typ „FileDoesNotExistException“ oder „FileNotReadableException“ ist bzw. von einer dieser Klassen abgeleitet ist. Und natürlich ungeprüfte Exceptions. • void h() Diese Funktion darf keine geprüften Exceptions werfen, sondern nur ungeprüfte. Achtung – die Exception-Spezifikation bezieht sich nur auf geprüfte Exceptions. Jede Funktion kann jederzeit ungeprüfte Exceptions werfen. Durch die Exception-Spezifikation weiß der Compiler, ob eine Funktion eine geprüfte Exception werfen könnte oder nicht. Der Funktions-Aufruf einer Funktion, die eine geprüfte Exception werfen könnte, muss seinerseits wieder entweder durch einen try-catch-Block oder eine Exception-Spezifikation gesichert sein. public void f1() throws Exception { throw new Exception(); } public void f2() { f1(); } // Compiler-Fehler, unbehandelte Exception Lösung 1 – Abfangen der Exception in der Funktion selber public void f1() throws Exception { throw new Exception(); } public void f2() { try { f1(); } catch (Exception x) } } // Exception wird abgefangen { Lösung 2 – Funktion mit Exception-Spezifikation public void f1() throws Exception { throw new Exception(); } public void f2() throws Exception { f1(); // Exception wird nach aussen gemeldet } Die Exception-Spezifikationen von Java sind nicht unumstritten: Ein klarer Vorteil ist, dass - zumindest die geprüften Exceptions - im Code behandelt werden müssen, da der Compiler dies prüft. Für die geprüften Exceptions können daher zur Laufzeit keine unangenehmen Überraschungen auftreten. In der Praxis findet man aber häufig try-Blöcke mit leeren catch-Blöcken, d.h. ohne wirkliche Fehlerbehandlung. Diese Blöcke stehen dann nur da, um den Compiler zu befriedigen, nicht aus wirklicher Überzeugung. Hier im Tutorial z.B. finden sich einige Beispiele mit leeren catch© Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 387 / 409 Blöcken - zum Teil da noch keine Exceptions bekannt waren, und zum Teil da das Beispiel um andere Sprach-Features ging und der Code nicht zu sehr aufgebläht werden sollte. Exception-Spezifikation blähen die Funktions-Signatur häufig um technische Fehler auf, die auf unteren Ebenen nicht behandelt werden können. Z.B. Netzwerk- oder Datenbank-Probleme sind typische Beispiele. Die entsprechenden Exceptions sind geprüfte Exceptions. Da sie auf unteren Ebenen nicht sinnvoll behandelt werden können, müssen sie nach oben durchgereicht werden, und das heißt wiederrum lange Exception-Spezifikationen ohne wirklichen Vorteil denn das eine Datenbank-Operation schief gehen kann, sollte auch so jedem klar sein. In der Praxis geht daher mittlerweile die Tendenz zu ungeprüften Exceptions. So gibt es z.B. im Serverbereich das immer beliebtere Framework „Spring“, das u.a. mit dem Hauptvorteil wirbt, dass es die geprüften JDBC (Datenbank) Exceptions automatisch in ungeprüfte überführt. Die größte Gefahr ist aber das fehlende Bewußtsein vieler Programmierer für ungeprüfte Exceptions. Aufgrund der immer wieder vorkommenden Compiler-Fehler wegen nicht behandelter geprüfter Exceptions entsteht bei vielen Programmieren das Gefühl, sobald der Compiler nicht mehr nervt habe man sich um alle möglichen Probleme gekümmert. Dabei wird dann vergessen, dass es viele - sehr viele - ungeprüfte Exceptions gibt, die mögicherweise nicht bedacht worden sind. Hierbei sollte auch bedacht werden, dass viele sehr elementare Exceptions wie z.B. die „NullPointerException“ oder auch die „ArrayIndexOutOfBoundsException“ zu den ungeprüften Exceptions gehören. Und sehr kritische Fehler, wie z.B. fehlender Speicher, werden auch über ungeprüfte Exceptions gemeldet für die dann auch noch die Empfehlung ausgesprochen wird „sich nicht drum zu kümmern, da man in Java dann eh nichts mehr machen kann“ - siehe Kapitel todo. Ein weiterer wichtiger Verweis ist auch das Kapitel todo über das Thema „Exception-Sicherheit“. 22.10 Verschachtelte try-Blöcke Natürlich kann in einem try-Block wieder ein try-Block verkommen – direkt innerhalb der Funktion (Bsp. 1) oder auch indirekt durch Funktions-Aufrüfe (Bsp. 2). public static void main(String[] args) { try { System.out.println("Aeusserer try-Block"); try { System.out.println("Innerer try-Block"); throw new Exception(); } catch (Exception x) { System.out.println("Innerer catch-Block"); } System.out.println("Aeusserer try-Block Ende"); } catch (Exception x) { System.out.println("Aeusserer catch-Block"); } } private static void f() { try { System.out.println("Innerer try-Block"); throw new Exception(); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 388 / 409 } catch (Exception x) { System.out.println("Innerer catch-Block"); } } public static void main(String[] args) { try { System.out.println("Aeusserer try-Block"); f(); System.out.println("Aeusserer try-Block Ende"); } catch (Exception x) { System.out.println("Aeusserer catch-Block"); } } Ausgabe (in beiden Beispielen identisch) Aeusserer try-Block Innerer try-Block Innerer catch-Block Aeusserer try-Block Ende In der Praxis ist gerade die zweite Variante nicht ungewöhnlich, da man häufig nicht weiß was in den aufgerufenden Funktionen genau passiert. Und das intern irgendwas schief gehen könnte, ist recht normal. Drei Anwendungen sind hierbei besonders interessant: • Erneutes Auswerfen einer Exception • Auswerfen einer anderen Exception • Unbehandelte Exceptions im inneren Block 22.10.1 Erneutes Auswerfen einer Exception Eine gefangene Exception kann natürlich jederzeit im catch-Block wieder geworfen werden. Dies passiert, wenn man auf einen Fehler oder eine Fehler-Gruppe dediziert reagieren möchte, den Fehler aber nicht endgültig behandeln kann – ihn also weitermelden muß. public static void main(String[] args) { try { System.out.println("Aeusserer try-Block"); try { System.out.println("Innerer try-Block"); throw new Exception(); } catch (Exception x) { System.out.println("Innerer catch-Block"); throw x; } // System.out.println("Aeusserer try-Block Ende"); } catch (Exception x) { System.out.println("Aeusserer catch-Block"); } } unreachable code Ausgabe Aeusserer try-Block Innerer try-Block Innerer catch-Block © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 389 / 409 Aeusserer catch-Block Hinweis – mit Verlassen des catch-Blocks gilt die primäre Exception als behandelt. Die neu geworfene wurde aber noch nicht gefangen – und gilt daher nun als unbehandelt. Hierbei ist es unwichtig, dass das neu geworfene Exception-Objekt identisch zum alten ist. 22.10.2 Auswerfen einer anderen Exception Statt der Original Exception kann auch eine beliebige andere Excepton geworfen werden. Dies wird häufig gemacht, wenn die Original Exception nicht nach aussen bekannt gemacht werden soll, da sie z.B. aus einer externen Bibliothek kommt. public static void main(String[] args) { try { System.out.println("Aeusserer try-Block"); try { System.out.println("Innerer try-Block"); throw new Exception(); } catch (Exception x) { System.out.println("Innerer catch-Block"); throw new RuntimeException(); } // System.out.println("Aeusserer try-Block Ende"); } catch (Exception x) { System.out.println("Aeusserer catch-Block"); } } unreachable code Ausgabe Aeusserer try-Block Innerer try-Block Innerer catch-Block Aeusserer catch-Block Hinweis – man nennt dies auch Exception-Mapping. Hinweis – mit Verlassen des catch-Blocks gilt die primäre Exception als behandelt. Die neu geworfene wurde aber noch nicht gefangen – und gilt daher nun als unbehandelt. 22.10.3 Unbehandelte Exceptions im inneren Block Es ist nicht notwendig, dass die catch-Blöcke eines try-Blocks alle möglichen Exceptions fangen. Ist kein passender catch-Block vorhanden, wird der Stack weiter abgebaut bis ein trymit einem passenden catch-Block gefunden wird. public static void main(String[] args) { try { System.out.println("Aeusserer try-Block Start"); try { System.out.println("Innerer try-Block Start"); System.out.println("Werfe Exception"); throw new Exception(); } catch (ClassCastException x) { System.out.println("Innerer catch-Block: ClassCastException gefangen"); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 390 / 409 } System.out.println("Aeusserer try-Block Ende"); } catch (Exception x) { System.out.println("Aeusserer catch-Block: Exception gefangen"); } } Ausgabe Aeusserer try-Block Start Innerer try-Block Start Werfe Exception Aeusserer catch-Block: Exception gefangen Dies ist ein ganz normaler Fall. Sie müssen nicht an jeder Stelle alle möglichen Exceptions abfangen, nur weil sie geworfen werden könnten. Sondern an den Stellen, an denen sie bestimmte Fehler behandeln, sie melden, neu aufsetzen oder sonst was können – also sinnvoll auf diese Fehler reagieren können – an denen fangen sie sie ab. Ansonsten ignorieren sie sie – im Rahmen der Exception-Sicherheit, siehe Kapitel 22.12. 22.11 Exception-Objekte Ein Exception sollte detailierte Informationen über den Fehler enthalten, der zu ihrem Werfen führte. Dann können Meldungen ausgeben werden, die helfen den Fehler zu finden und zu beheben. Da sie prinzipiell für jeden Fehlertyp eine eigene Exception-Klasse entwerfen können, stehen ihnen die kompletten Möglichkeiten einer Klasse zur Verfügung um diese Informationen zu transportieren. Unabhängig davon erbt jedes Exception Objekt, da es direkt oder indirekt von „Throwable“ abgeleitet ist, mehrere Funktionen – und damit implizit Funktionalitäten. Zwei davon möchte ich hier vorstellen: • public String getMessage() Gibt einen String zurück, der die Exception näher beschreibt. • public void printStackTrace() Gibt die aktuelle Exception und den Funktions-Stack zum Zeitpunkt ihrer Erzeugung auf der Error-Ausgabe aus. public class Appl { private static void f3() { throw new RuntimeException("Ich bin ein Fehler-Objekt"); } private static void f2() { f3(); } private static void f1() { f2(); } public static void main(String[] args) { try { f1(); } catch (Throwable t) { © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 391 / 409 System.out.println("Throwable gefangen"); System.out.println("Obj: \"" + t + '"'); System.out.println("Msg: \"" + t.getMessage() + '"'); System.out.println(); t.printStackTrace(); } } } Ausgabe Throwable gefangen obj: "java.lang.RuntimeException: Ich bin ein Fehler-Objekt" msg: "Ich bin ein Fehler-Objekt" java.lang.RuntimeException: Ich bin ein Fehler-Objekt at Appl.f3(Appl.java:4) at Appl.f2(Appl.java:8) at Appl.f1(Appl.java:12) at Appl.main(Appl.java:17) Hinweis – diesmal ist das Beispiel der vollständige Quelltext, damit sie sehen das im StackTrace Dateiname und Zeilennummern der Funktions-Aufrufe enthalten sind. 22.12 Exception-Sicherheit So einfach und schlüssig Exceptions im ersten Augenblick auch wirken - es ist schwer, wirklich Exception-sicheren Code zu schreiben. Exception-sicherer Code ist Code, der im Falle einer Exception keinen undefinierten Status zurückläßt, sondern die Objekte immer in einem sauberen Zustand hinterläßt. Hierbei werden drei Level unterschieden: Level 1 – Starke-Garantie Wenn eine Exception geworfen wird, bleibt der Status des Programms im logischen Sinne unverändert. Diese Garantie impliziert eine globale Commit/Rollback Strategie, die auch dafür sorgt, dass z.B. Referenzen und Iteratoren nach einer Exception noch korrekt sind. Level 2 – Mindest-Garantie Wenn eine Exception geworfen wird, so entstehen keine Ressourcen-Löcher (z.B. nicht geschlossene Dateien oder Netzwerk-Verbindungen), und alle Objekte bleiben konsistent und benutzbar. Dies ist die Mindest-Garantie, die gegeben sein muss, damit ein Programm funktionsfähig bleibt. Level 3 – keine Garantie Wenn eine Exception geworfen wird, so entstehen Ressourcen-Löcher bzw. nicht benutzbare oder inkonsistente Objekte. Level 3 darf nie passieren, denn dann haben sie ein potentielles Problem in ihrem Programm. Level 1 wäre der Ideal-Zustand, der aber häufig nicht erreichbar ist, aber angesprebt werden sollte. Hier ein kleines Beispiel – Exception-sicher oder nicht? © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 392 / 409 public class Person { private String name; private int age; private String phone; public Person(String n, String a, String p) { name = n; age = Integer.parseInt(a); phone = p; } public String toString() { return name + " (Alter: " + age + ") - Telefon: " + phone; } public void init(String n, String a, String p) { name = n; age = Integer.parseInt(a); phone = p; } } Natürlich nicht! Schauen wir uns z.B. folgende Benutzung an: public static void main(String[] args) { Person prs = new Person("Detlef", "23", "0241 / 123 456"); System.out.println("Person: " + prs); try { prs.init("Bernd", "--", "012 / 345 67"); } catch (Exception x) { } System.out.println("Person: " + prs); } Ausgabe Person: Detlef (Alter: 23) - Telefon: 0241/ 123 456 Person: Bernd (Alter: 23) - Telefon: 0241/ 123 456 Wie sie an der Ausgabe sehen, existiert nach der Exception ein Personen-Objekt, das einen inkonsistenten Zustand hat. Denn eine Person „Bernd“ mit Alter „23“ und der Telefon-Nr. „0241/123456“ gibt es nicht. Das Problem ist hier die „init“ Funktion in der Klasse „Person“. Mitten in der Veränderung der Attribute der Klasse kann eine Exception geworfen werden – in diesem Augenblick ist ein Teil der Attribute verändert, während ein anderer Teil noch den alten Wert hat. Das darf nicht passieren! Die Lösung ist entweder: • Erst alle Veränderungen lokal machen, und sie dann in einem Rutsch Exception-sicher ausführen. Oder • Alle Veränderungen Stück für Stück ausführen, aber im Falle einer Exception die schon gemachten rückgängig zu machen. public void init(String n, String a, String p) { int ag = Integer.parseInt(a); name = n; age = ag; © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 393 / 409 phone = p; } Das Problem ist, dass an sehr vielen Stellen Exceptions geworfen werden können – man aber nicht daran denkt, da man sie im normalen Code nicht sieht. Und da viele Exceptions keine geprüften Exceptions sind, macht einen der Compiler auch nicht darauf aufmerksam. 23 Streams 23.1 Einführung Ein allgemeines Konzept in der Software-Entwicklung sind die sogenannte Streams. Ein Stream ist eigentlich nicht mehr als ein Datenstrom, der die Daten von irgendwoher bezieht und irgendwohin liefert. Außerdem können Streams möglicherweise manipuliert werden (z.B. geleert werden). Abb. 23-1 : Ein nicht näher spezifizierter Datenstrom In der Praxis sind zwei Stream-Varianten besonders wichtig: • Byte-Streams, d.h. Streams, bei denen der Datenstrom aus einfachen Bytes ohne jede Struktur besteht. Byte-Streams werden in Java durch Input-Streams bzw. Output-Streams repräsentiert – siehe Kapitel todo. • Zeichen-Streams, d.h. Streams, bei denen der Datenstrom aus Zeichen besteht. Da Java intern mit Unicode UTF-16 arbeitet, sind hier im Normallfall 2 Byte große Zeichen mit UTF16 Codierung gemeint. Zeichen-Streams werden in Java durch Reader und Writer repräsentiert – siehe Kapitel todo. Bemerkung – da Streams sehr allgemein sind, werden sie oft als Grundlage für ein Kommunikation-Konzept zwischen zwei beliebigen „Partnern“ genommen, z.B. zwischen Prozessen, Threads, zwei Netzwerk-Teilnehmern, einem Datei-System und einem Programm, usw. 23.2 Java Streams 23.2.1 Architektur In Java stehen 4 Stream-Hierarchien zur Verfügung: • Eingabe Byte-Streams – abstrakte Basis-Klasse: „java.io.InputStream“ • Ausgabe Byte-Streams – abstrakte Basis-Klasse: „java.io.OutputStream“ • Eingabe Zeichen-Streams – abstrakte Basis-Klasse: „java.io.Reader“ • Ausgabe Zeichen-Streams – abstrakte Basis-Klasse: „java.io.Writer“ Warum gibt es überhaupt 4 Stream-Hierarchien, und nicht einfach 4 Stream-Klassen? © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 394 / 409 • Nun, im einfachsten Fall transportieren Streams ihre Daten unverändert von der Quelle zum Ziel. Aber es gibt auch Streams, die die Daten unterwegs manipulieren – z.B. puffern, filtern, zählen, zippen bzw. unzippen, Hash-Werte bestimmen, und vieles mehr. • Außerdem sind die Quellen und Ziele je nach Anforderung unterschiedlich. In einem Fall wird z.B. aus einer Datei gelesen und in einen String geschrieben. In einem anderen Fall aus einem Byte-Array gelesen und in eine Datei geschrieben. Statt einer Klasse, die „alles“ anbietet, wurde in Java die bessere und offenere Architektur einer Klassen-Hierarchie gewählt. • Es gibt eine abstrakte Basis-Klasse, z.B. „InputStream“ • Pro Verhalten gibt es eine konkrete abgeleitete Klasse, z.B. „BufferedInputStream“. • Viele Streams können einen anderen gleichartigen Stream kapseln. Welche Vorteile hat eine solche Architektur? • Kleine, übersichtliche und gekapselte Komponenten Jede einzelne Klasse ist nur für ein Verhalten da, und kann so klein und übersichtlich – auch bzgl. ihrer Schnittstelle – gehalten werden. • Erweiterbarkeit und Offenheit Implementiert keine Stream-Klasse ein gewünschtes Verhalten, so kann es jederzeit selber implementiert und in die Stream-Verarbeitung integriert werden. • Kombinierbarkeit Da viele Streams ineinander gekapselt werden können, können die einzelnen Transformationen in beliebigen Kombinationen und Reihenfolgen aufeinander angewandt werden. • Erweiterbarkeit, Offenheit und Kombinierbarkeit Es können alle Quellen mit allen Zielen kombiniert werden, und es können eigene Quellen bzw. Ziele in die Stream-Verarbeitung integriert werden. Gerade die Voraussetzung „Streams fast beliebig ineinander zu kapseln“ erlaubt eine wahre Vielfalt an Lösungen. So könnte z.B. ein Input-Stream eine Datei kapseln. Um diesen gepuffert auszulesen, wird er in einem gepuffert-Input-Stream gekapselt. Ist die Datei verschlüsselt, würde man diesen in einem Entschlüsselungs-Stream kapseln. Und wenn der eigentliche Inhalt dann noch gezippt ist, dann wird noch ein Unzip-Input-Stream um diesen herum gelegt. Abb. 23-2 : Bsp für ineinander gekapselte Streams Hinweis – funktionieren tut dies intern natürlich wieder mit Vererbung und Polymorphie. Da z.B. alle Input-Streams von einer Basis-Klasse abgeleitet sind, können sie alle über diese gehandelt werden. Da sie die Schnittstelle der Basis-Klasse überschrieben haben, werden die Aufrufe via Polymorphie transparent an die jeweiligen Implementierungen weitergereicht. Abb. 23-3 : Arbeitsweise der Stream-Kapselung © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 395 / 409 23.2.2 Byte-Streams und Zeichen-Streams Der allgemeinere Stream-Typ ist natürlich der Byte-Stream, da intern alle Daten aus Bytes aufgebaut sind. Um aus einem Byte-Stream Zeichen zu lesen, existiert daher die von „Reader“ abgeleitete Klasse „java.io.InputStreamReader“, die einen Input-Stream in einem Reader kapselt. Umgekehrt gibt es die von „Writer“ abgeleitete Klasse „java.io.OutputStreamWriter“, die einen Output-Stream in einem Writer kapselt. Ein Beispiel hierfür findet sich z.B. in Kapitel 23.3.1. 23.2.3 Stream-Hierarchien Viele spezielle Stream-Fähigkeiten (d.h. konkrete Stream-Klassen) finden sich in allen 4 Stream-Hierarchien. Zum einen werden natürlich für die meisten Eingabe-Streams auch die entsprechenden Gegenstücke auf der Ausgabe-Seite benötigt, bzw. umgekehrt. Zum anderen sind viele Fähigkeiten unabhängig vom Daten-Typ (d.h. Byte oder Zeichen), und sind daher in beiden Hierarchien sinnvoll. Stellvertretend für alle Stream-Hierarchien wird hier die Eingabe Byte-Stream Hierarchie detailierter betrachtet. Für die anderen Streams wird auf die Java-Hilfe verwiesen. java.lang.Object java.io.InputStream (abstract) – siehe auch Kapitel todo javax.sound.sampled.AudioInputStream java.io.ByteArrayInputStream – siehe auch Kapitel todo java.io.FileInputStream – siehe auch Kapitel todo java.io.FilterInputStream java.io.BufferedInputStream – siehe auch Kapitel todo java.util.zip.CheckedInputStream javax.crypto.CipherInputStream java.io.DataInputStream – siehe auch Kapitel todo java.security.DigestInputStream java.util.zip.InflaterInputStream java.util.zip.GZIPInputStream java.util.zip.ZipInputStream java.util.jar.JarInputStream java.io.LineNumberInputStream (deprecated) javax.swing.ProgressMonitorInputStream java.io.PushbackInputStream – siehe auch Kapitel todo org.omg.CORBA.portable.InputStream java.io.ObjectInputStream – siehe auch Kapitel todo java.io.PipedInputStream java.io.SequenceInputStream – siehe auch Kapitel todo java.io.StringBufferInputStream (deprecated) © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 396 / 409 23.2.3.1 Klasse „java.io.InputStream“ Basis-Klasse für alle byte-orientierten Input-Streams • public int available() throws IOException • public void close() throws IOException • public abstract int read() throws IOException • public int read(byte[ ] b) throws IOException 23.2.3.2 Klasse „java.io.ByteArrayInputStream“ Hiermit kann ein Byte-Array mit einem Input-Stream verbunden werden. byte[] array = new byte[10]; ByteArrayInputStream in = new ByteArrayInputStream(array); 23.2.3.3 Klasse „java.io.FileInputStream“ Ist ein Stream, der mit einem File verbunden wird. • public FileInputStream(File file) throws FileNotFoundException • public FileInputStream(String name) throws FileNotFoundException Ein Beispiel hierfür finden sie z.B. in Kapitel 23.3.2. 23.2.3.4 Klasse „java.io.BufferedInputStream“ Implementiert einen gepufferten Input-Stream • public BufferedInputStream(InputStream in) Ein Beispiel hierfür finden sie z.B. in Kapitel 23.3.2 und Kapitel 25. 23.2.3.5 Klasse „java.io.DataInputStream“ Mit diesem Stream können die elementaren Datentypen aus einem zugrunde liegenden InputStream auf maschinenunabhängge Weise gelesen werden. • public DataInputStream(InputStream in) • public final int readInt() throws IOException 23.2.3.6 Klasse „java.io.PushbackInputStream“ Implementiert einen Stream mit Puffer, in den Daten zurückgeschrieben werden können. • public PushbackInputStream(InputStream in) • public void unread(byte[ ] b) throws IOException 23.2.3.7 Klasse „java.io.ObjectInputStream“ Implementiert einen Stream, aus dem komplette Objekte heraus gelesen werden können. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 397 / 409 • public ObjectInputStream(InputStream in) throws IOException, ... • public final Object readObject() throws OptionalDataException, ClassNotFoundException Beispiele hierfür finden sich im Kapitel über Serialisierung – siehe Kapitel 25. 23.2.3.8 Klasse „java.io.SequenceInputStream“ Kann Input-Streams sequenziell koppeln. • public SequenceInputStream(InputStream s1, InputStream s2) Ein Beispiel hierfür finden sie z.B. in Kapitel 23.3.4. 23.3 Beispiele 23.3.1 Beispiel „Tastatur-Eingabe“ Beispiel aus Kapitel 3.10. Erläuterung – todo... public static void main(String[] args) { try { System.out.println("Echo-Programm"); InputStreamReader isr = new InputStreamReader(System.in); BufferedReader reader = new BufferedReader(isr); while (true) { System.out.print("> "); String in = reader.readLine(); if (in.length()==0) { break; } System.out.println(" \"" + in + "\" - " + in.length() + " Zeichen"); } } catch (Exception x) { } System.out.println("Programm-Ende"); } mögliche Ausgabe Echo-Programm > Hallo "Hallo" - 5 Zeichen > Ich lerne jetzt Java "Ich lerne jetzt Java" - 20 Zeichen > Programm-Ende 23.3.2 Beispiel „Datei lesen“ Erläuterung – todo... public static void main(String[] args) { try { System.out.println("Lese Datei \"Testdaten.txt\" - Version 1"); File file = new File("Testdaten.txt"); FileInputStream fis = new FileInputStream(file); InputStreamReader isr = new InputStreamReader(fis); © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 398 / 409 BufferedReader reader = new BufferedReader(isr); while (true) { String in = reader.readLine(); if (in==null) break; System.out.println("> " + in); } System.out.println("Datei-Ende"); } catch (Exception x) { System.out.println("Probleme beim Lesen der Datei"); } } mögliche Ausgabe Lese Datei "Testdaten.txt" > Hallo Welt, > > diese Zeilen kommen aus der Datei. Datei-Ende Eine alternative kürzere Variante arbeitet mit der Klasse „java.io.FileReader“, die intern ein FileObjekt erzeugt, dieses in einen File-Input-Stream und dann in einem Input-Stream-Reader kapselt. public static void main(String[] args) { try { System.out.println("Lese Datei \"Testdaten.txt\" - Version 2"); FileReader fr = new FileReader("Testdaten.txt"); BufferedReader reader = new BufferedReader(fr); while (true) { String in = reader.readLine(); if (in==null) break; System.out.println("> " + in); } System.out.println("Datei-Ende"); } catch (Exception x) { System.out.println("Probleme beim Lesen der Datei"); } } 23.3.3 Beispiel „Lesen mit Zeilen-Nummer“ Erläuterung – todo... public static void main(String[] args) { try { System.out.println("Echo-Programm mit Zeilen-Nummern"); InputStreamReader isr = new InputStreamReader(System.in); LineNumberReader reader = new LineNumberReader(isr); while (true) { System.out.print("> "); String in = reader.readLine(); if (in.length() == 0) { break; } System.out.print(reader.getLineNumber() + ":"); System.out.println(" \"" + in + "\" - " + in.length() + " Zeichen"); } } catch (Exception x) { } } © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 399 / 409 mögliche Ausgabe Echo-Programm mit Zeilen-Nummern > Hallo Welt, 1: "Hallo Welt," - 11 Zeichen > ich lerne jetzt Java. 2: "ich lerne jetzt Java." - 21 Zeichen > Bald schon kann ich tolle Sachen schreiben. 3: "Bald schon kann ich tolle Sachen schreiben." - 43 Zeichen > Bis bald... 4: "Bis bald..." - 11 Zeichen > 23.3.4 Beispiel „Sequenzielle Kopplung“ Erläuterung – todo... public static void main(String[] args) { try { System.out.println("Lesen aus zwei gekoppelten Strings:"); String s1 = "Zeile 1\n\nUnd dies ist "; String s2 = "die Zeile 3\nZeile 4"; // StringBufferInputStream ist zwar deprecated, aber die Klasse // ermoeglicht das einfachste Beispiel - darum doch. StringBufferInputStream sbis1 = new StringBufferInputStream(s1); StringBufferInputStream sbis2 = new StringBufferInputStream(s2); SequenceInputStream sis = new SequenceInputStream(sbis1, sbis2); InputStreamReader isr = new InputStreamReader(sis); LineNumberReader reader = new LineNumberReader(isr); while (true) { String in = reader.readLine(); if (in==null) { break; } System.out.println(reader.getLineNumber() + ": \"" + in + "\""); } } catch (Exception x) { } } Ausgabe Lesen aus zwei gekoppelten Strings: 1: "Zeile 1" 2: "" 3: "Und dies ist die Zeile 3" 4: "Zeile 4" 23.4 Stream-Tokenizer Hilfs-Klasse um Zeichen-Ströme in Token zu zerlegen: „java.io.StreamTokenizer“. • public StreamTokenizer(Reader r) • public int ttype • public static final int TT_EOF • public static final int TT_EOL • public static final int TT_NUMBER • public static final int TT_WORD • public String sval © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 400 / 409 • public double nval • public int nextToken() throws IOException public static void main(String[] args) { try { String str = "3 mal 3 ist ach..., aeh ... 9!"; StringReader sr = new StringReader(str); StreamTokenizer st = new StreamTokenizer(sr); while (st.nextToken() != StreamTokenizer.TT_EOF) { switch (st.ttype) { case StreamTokenizer.TT_EOL: System.out.println("End of Line"); break; case StreamTokenizer.TT_NUMBER: System.out.println("Zahl: " + st.nval); break; case StreamTokenizer.TT_WORD: System.out.println("Wort: " + st.sval); break; } } } catch (Exception x) { } } Ausgabe Zahl: 3.0 Wort: mal Zahl: 3.0 Wort: ist Wort: ach... Wort: aeh Zahl: 0.0 Zahl: 0.0 Zahl: 0.0 Zahl: 9.0 Der Case-Fall „StreamTokenizer.TT_EOL“ wird hier eigentlich nicht benötigt, da defaultmässig im StringTokenizer ein EOL ein Whitespace ist. 24 Reflexion Java bietet ein Sprachmittel, mit dem zur Laufzeit Informationen über beliebige Objekte abgefragt und genutzt werden können. Dazu existiert in Java die Klasse „java.lang.Class“, die eine Art Meta-Ebene über den normalen Klassen darstellt, d.h. sie enthält Informationen über die Klassen selber. Mit solchen Informationen kann man anders programmieren, als man es von ‘normalen’ Compiler-Sprachen gewöhnt ist. 24.1 Objekt-Erzeugung über den Klassen-Namen Eine der Möglichkeiten die sich durch solche Meta-Informationen bietet, ist die Erstellung von Objekten einer Klasse zur Laufzeit nur unter Angabe des Klassen-Namens. Bitte verstehen Sie diesen Satz richtig. Es geht nicht um die normale Erzeugung eines Objekts mit „new“ in der Art: © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 401 / 409 String s = new String(); sondern um eine Erzeugung, bei der der Klassen-Name zur Compile-Zeit noch vollkommen unbestimmt ist. Object o1 = ClassCreator.create("KlassenName"); Object o2 = ClassCreator.create(string); Hierbei erzeugt die Funktion „create“ der Klasse „ClassCreator“ zur Laufzeit aus dem String ein Objekt der Klasse und liefert dieses zurück. Der String kann z. B. auch aus einer BenutzerEingabe oder einer Datenbank87 stammen, d.h. er ist zur Compile-Zeit vollkommen unbestimmt. Die Klasse „ClassCreator“ und die Funktion „create“ sind hierbei willkürlich gewählt – sie existieren so in Java nicht. Es ging dabei erstmal nur darum, den obigen Code einfach zu halten, und den Unterschied zu „new“ zu verdeutlichen. Aber wir können problemlos eine Klasse erstellen, die sich entsprechend verhält. public class ClassCreator { public static Object create(String name) { try { Class descriptor = Class.forName(name); return descriptor.newInstance(); } catch (Throwable x) { System.out.println("Probleme in ClassCreator.create(String)"); } return null; } } Und so wird sie dann benutzt: StringBuffer s = (StringBuffer)ClassCreator.create("java.lang.StringBuffer"); s.append("Java ganz meta-maessig"); System.out.println(s); Ausgabe Java ganz meta-maessig „Object forName(String)“ ist eine Klassen-Funktion der Klasse „Class“, die einen String - den vollständig referenzierten Klassen-Namen - erwartet, und dann einen Klassen-Descriptor vom Typ „Class“ zurückgibt. Dieses Objekt der Klasse „Class“ enthält eine vollständige Beschreibung der angegebenen Klasse. Mit diesem Descriptor kann z.B. mit „newInstance()“ ein neues Objekt der Klasse erzeugt werden88. Achtung – die Klasse des so erzeugten Objekts muss einen Standard-Konstruktor haben, da die JVM keine Parameter herbeizaubern kann. Der Mechanismus der Serialisierung (siehe Kapitel todo...) arbeitet genau mit diesem Trick. 88 Da ClassCreator.create ein Object zurückgibt, muss das Ergebnis natürlich gecastet werden, damit es benutzt werden kann. 87 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 402 / 409 Folgende Probleme können auftauchen: String Leerstring "", bzw. Unerlaubter Name Unreferenzierter Klassen-Name Nicht public Klasse in fremdem Package Klasse mit privatem Standard-Konstruktor Klasse ohne Standard Konstruktor Abstrakte Klasse bzw. Interface Exception IllegalArgumentException ClassNotFoundException IllegalAccessException IllegalAccessException NoSuchMethodError IllegalArgumentException 24.2 Klasse „Class“ Für jede Klasse in der virtuellen Maschine wird ein „Class“ Objekt angelegt. Mit der Funktion ‘Class getClass()’ der Klasse „Object“ kann man daher direkt an die Meta-Informationen zu einem Objekt gelangen.89 String s = "Hallo"; Class c = s.getClass(); 24.2.1 Element-Funktionen Die Klasse „Class“ enthält Element-Funktionen, mit der Information über die Klasse des zugehörigen Class-Objekts zur Laufzeit ermittelt werden können. Hier eine kleine Auswahl: Funktion String getName() Constructor[ ] getConstructors() Constructor[ ] getDeclaredConstructors() Method[ ] getMethods() Method[ ] getDeclaredMethods() Field[ ] getFields() Field[ ] getDeclaredFields() boolean isArray() boolean isInterface() boolean isPrimitive() Rückgabewert voll referenziert Name der Klasse Array aller public Konstruktoren Array aller Konstruktoren Array aller public Methoden Array aller Methoden Array aller public Attribute Array aller Attribute ob das Objekt ein Array ist ob das Objekt ein Interface ist ob das Objekt ein elementarer Datentyp ist Hinweis – alle diese Klassen (z.B. „Constructor“, „Method“,...) und viele weitere kommen aus dem Package „java.lang.reflect“. Und hier ein kleines einfaches Beispiel: class Reflect { private int privateInt; Da in Java jede Klasse immer direkt oder indirekt von Object abgeleitet ist, ist diese Funktion immer vorhanden. 89 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 403 / 409 protected boolean protectedBoolean; public double publicDouble; public Reflect() {} public Reflect(int i) {} private void privateFct() {} protected void protectedFct() {} public void publicFct() {} public int fct1() { return 0; } public void fct1(int i) {} public void fct1(int i, String s) {} } public static void main(String[] args) { Reflect r = new Reflect(); Class c = r.getClass(); System.out.println("Allgemeine Klasseninformationen"); System.out.println(" " + c.getName()); System.out.println("Konstruktoren"); Constructor[] con = c.getConstructors(); for (int i=0; i<con.length; i++) { System.out.println(" " + con[i]); } System.out.println("Methoden"); Method[] m = c.getMethods(); for (int i=0; i<m.length; i++) { System.out.println(" " + m[i]); } System.out.println("Attribute"); Field[] f = c.getDeclaredFields(); for (int i=0; i<f.length; i++) { System.out.println(" " + f[i]); } } Ausgabe Allgemeine Klasseninformationen Reflect Konstruktoren public Reflect(int) public Reflect() Methoden public void Reflect.publicFct() public void Reflect.fct1(int) public void Reflect.fct1(int,java.lang.String) public int Reflect.fct1() public native int java.lang.Object.hashCode() public final native java.lang.Class java.lang.Object.getClass() public final void java.lang.Object.wait(long,int) throws java.lang.InterruptedException public final void java.lang.Object.wait() throws java.lang.InterruptedException public final native void java.lang.Object.wait(long) throws java.lang.InterruptedException public boolean java.lang.Object.equals(java.lang.Object) public final native void java.lang.Object.notify() public final native void java.lang.Object.notifyAll() public java.lang.String java.lang.Object.toString() Attribute private int Reflect.privateInt protected boolean Reflect.protectedBoolean public double Reflect.publicDouble © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 404 / 409 Die Parameter-Namen z.B. lassen sich übrigens zur Laufzeit nicht mehr feststellen, da diese Informationen nicht zur Klasse gehören, und vom Compiler nirgendwo hinterlegt werden. 24.3 Disassemblieren Bei dem Thema Reflexion liegt die Frage nahe, ob jemand bestehenden Java Byte-Code decompilieren bzw. disassemblieren kann, und damit an das Know-How des Entwicklers / der Firma kommen kann, der / die diesen Code geschrieben hat. Die komplette Programmstruktur liegt immer im Java Byte-Code vor, sonst könnte das Reflexion API nicht funktionieren. Ausserdem kann der Byte Code in hohen Masse wieder in relativ lesbaren Quellcode zurückgewandelt werden, denn: • Die Abbildung von Anweisungen auf Code ist relativ eindeutig. • Es gibt in Java keinen Präprozessor. • OO Funktion sind meist sehr klein (90% weniger als 20 Zeilen) • Der Compiler kann nur wenige Optimierungen machen. Nur die Namen der Parameter und der lokalen Variablen fehlen. Ein relativ einfacher Disassembler liegt dem Java SDK bei – es ist das Kommandozeilen-Tool „javap“. Es gibt viele viel leistungsfähigerer Disassembler – einen relativ guten Ruf hat das Tool „Jad“ – siehe http://www.kpdus.com/jad.html. Es gibt daher Tools – sogenannte Obfuscator – die sämtliche Symbol-Definitionen und deren Benutzungen mit unleserlichen Namen ersetzen. Dies funktioniert aber nur, solange keine Reflection eingesetzt wird. 24.4 Aufgaben 24.4.1 Aufgabe „Klassen-Inspektor“ Schreiben sie ein kleines grafisches Tool, einen Klassen-Inspektor. Hierbei kann der Benutzer einen vollständig referenzierten Klassen-Namen eingeben, und erhält danach eine Tabelle der: - (public) Konstruktoren - (public) Funktionen - (public) Attribute - usw. Lösung siehe Kapitel 24.5. 24.5 Lsg. zu Aufgabe „Klassen-Inspektor“ – Kap. 24.4.1 todo... © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 405 / 409 25 Serialisierung Der Mechanismus der Serialisierung und der Deserialisierung erlaubt es den Status von Objekten in einen Stream zu schreiben, bzw. ihn daraus zu lesen, um diesen z.B. nach einem Programmlauf für den nächsten zu sichern90. Oder auch um Objekt-Stati von einem Rechner auf den anderen zu transportieren. Für diesen Mechanismus sind in Java folgende Elemente vorhanden: • Interface „java.io.Serializable“ • Klasse „java.io.ObjectOutputStream“ • Klasse „java.io.ObjectInputStream“ • Interface „java.io.Externalizable“ • Interface „java.io.ObjectOutput“ • Interface „java.io.ObjectInput“ Genau genommen werden die Interface’s Externalizable, ObjectOutput und ObjectInput in vielen Fällen gar nicht für eine Serialisierung bzw. Deserialisierung benötigt, da das DefaultVerhalten von Java für die meisten Zwecke ausreicht. Über die Implementierung dieser Interface’s kann das Default-Verhalten der Serialisierung verändert und angepaßt werden. Da Serialisierung und Deserialisierung Ein- bzw. Ausgabe-Operationen sind, die fehlschlagen können, können alle Methoden eine IOException bzw. davon abgeleitete Exception-Klassen werfen. Manche Funktionen werfen unter Umständen auch noch andere Exceptions. Dies sind geprüfte Exceptions – bei der Benutzung werden also immer entsprechende try/catch Blöcke oder Exception-Spezifikationen notwendig sein. 25.1 Interface Serializable Mit dem Serializable Interface bietet Java einen leistungsstarken Default Mechanismus für Serialisierung und Deserialisierung, der in vielen Fällen ausreicht. Damit eine Klasse serialisiert bzw. deserialisiert werden kann, muss sie nur vom Interface Serializable abgeleitet werden. Da Serializable keine Funktion deklariert, muss in der Serializable-Klasse nichts weiter getan werden. import java.io.*; public class SerialClass implements Serializable { private String s; private int i; public SerialClass(String s, int i) { this.s = s; this.i = i; } public void print() { System.out.println("Objekt von SerialClass mit: " + s + " - " + i); 90 Dies wird oft auch mit dem Begriff ‘Persistenz’ belegt. © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 406 / 409 } } System.out.println("Schreiben"); try { SerialClass obj1 = new SerialClass("Objekt 1", 42); SerialClass obj2 = new SerialClass("Objekt 2", 23); obj1.print(); obj2.print(); FileOutputStream file = new FileOutputStream("objekte.dat"); ObjectOutputStream oos = new ObjectOutputStream(file); oos.writeObject(obj1); oos.writeObject(obj2); oos.flush(); file.close(); } catch (Exception x) { System.out.println("Fehler beim Schreiben: " + x); } Ausgabe Schreiben Objekt von SerialClass mit: Objekt 1 - 42 Objekt von SerialClass mit: Objekt 2 - 23 In diesem Beispiel werden zwei Objekte der Klasse „SerialClass“ in die Datei "objekte.dat" serialisiert (abgespeichert). Um die beiden Objekte wieder einzulesen ist genauso viel Aufwand nötig: System.out.println("Lesen"); try { FileInputStream file = new FileInputStream("objekte.dat"); ObjectInputStream ois = new ObjectInputStream(file); SerialClass obj1 = (SerialClass) ois.readObject(); SerialClass obj2 = (SerialClass) ois.readObject(); file.close(); obj1.print(); obj2.print(); } catch (Exception x) { System.out.println("Fehler beim Lesen: " + x); } Ausgabe Lesen Objekt von SerialClass mit: Objekt 1 - 42 Objekt von SerialClass mit: Objekt 2 - 23 Hinweis – intern arbeitet dies natürlich mit Reflection, über das alle notwendigen Informationen für die JVM zur Verfügung stehen. 25.2 Rekursion Die Serialisierung ist nicht auf die primäre Klassen-Ebene beschränkt, sondern läuft rekursiv durch alle abhängigen Objekte und speichert den gesamten Objekt-Graphen, der zur vollständigen Wiederherstellung nötig ist. public class SerialClass1 implements Serializable { private String s; private int i; © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 407 / 409 private SerialClass2 sc2; public SerialClass1(String s, int i) { this.s = s; this.i = i; sc2 = new SerialClass2(s); } public void print() { System.out.println("Objekt von SerialClass1 mit: " + s + " - " + i); System.out.println("- Unter-Objekt: " + sc2); } } public class SerialClass2 implements Serializable { private String s; public SerialClass2(String s) { this.s = s; } public String toString() { return "SerialClass2 mit \"" + s + "\""; } } Ausgabe in Anlehnung an das Bsp aus Kapitel 25.1. Schreiben Objekt von SerialClass1 mit: Objekt 1 - 42 - Unter-Objekt: SerialClass2 mit "Objekt 1" Objekt von SerialClass1 mit: Objekt 2 - 23 - Unter-Objekt: SerialClass2 mit "Objekt 2" Lesen Objekt von SerialClass1 mit: - Unter-Objekt: SerialClass2 Objekt von SerialClass1 mit: - Unter-Objekt: SerialClass2 Objekt 1 - 42 mit "Objekt 1" Objekt 2 - 23 mit "Objekt 2" Bei einer solchen Rekursion müssen auch alle abhängigen Klassen serialisierbar sein. Ist eine Klasse dies nicht, so werden beim Speichern bzw. beim Lesen entsprechende Exceptions geworfen – siehe Beispiel. public class SerialClass implements Serializable { private String s; private int i; private NonSerialClass nsc; public SerialClass(String s, int i) { this.s = s; this.i = i; nsc = new NonSerialClass(s); } public void print() { System.out.println("Objekt von SerialClass mit: " + s + " - " + i); System.out.println("- Unter-Objekt: " + nsc); } } public class NonSerialClass { private String s; © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 408 / 409 public NonSerialClass(String s) { this.s = s; } public String toString() { return "NonSerialClass mit \"" + s + "\""; } } Ausgabe in Anlehnung an das Bsp aus Kapitel 25.1. Schreiben Objekt von SerialClass mit: Objekt 1 - 42 - Unter-Objekt: NonSerialClass mit "Objekt 1" Objekt von SerialClass mit: Objekt 2 - 23 - Unter-Objekt: NonSerialClass mit "Objekt 2" Fehler beim Schreiben: java.io.NotSerializableException: NonSerialClass Lesen Fehler beim Lesen: java.io.WriteAbortedException: writing aborted; java.io.NotSerializableException: NonSerialClass 25.3 Klassen-Variablen Klassen-Variablen, d.h. static Attribute, werden nicht serialisiert und auch nicht deserialisiert, da sie der Klasse und nicht einem einzelnen Objekt gehören. 25.4 Modifier „transient“ Für Attribute gibt es den Modifier transient, der dafür sorgt, dass dieses Attribut nicht serialisert bzw. deserialisiert wird. Hier wird im Beispiel aus Kapitel 25.1 die Klasse „SerialClass“ durch die fast identische Klasse „SerialClassTransient“ ersetzt – einziger Unterschied ist das das „int“ Attribut den Modifier „transient“ bekommen hat. public class SerialClassTransient implements Serializable { private String s; private transient int i; public SerialClassTransient(String s, int i) { this.s = s; this.i = i; } public void print() { System.out.println("Objekt von SerialClassTransi. mit: " + s + " - " + i); } } Ausgabe nach der Deserialisierung in Anlehnung an das Bsp aus Kapitel 25.1. Schreiben Objekt von SerialClassTransient mit: Objekt 1 - 42 Objekt von SerialClassTransient mit: Objekt 2 – 23 Lesen Objekt von SerialClassTransient mit: Objekt 1 - 0 Objekt von SerialClassTransient mit: Objekt 2 - 0 © Detlef Wilkening 1997-2016 http://www.wilkening-online.de Objektorientiertes Programmieren in Java - V. 29 Seite 409 / 409 Das „int“ Attribut „i“ wird nicht serialisiert und beim Deserialisieren auf den Defaultwert des Typs (hier bei int ‘0’) gesetzt. 25.5 Nicht serialisierbare Basis-Klassen Hat die serialisierbare Klasse eine nicht serialisierbar Basis-Klasse, so muss diese einen Standard-Konstruktor besitzen, der bei der Deserialisierung zur Konstruktion des BasisKlassen-Anteils aufgerufen wird. Ist kein Standard-Konstruktor in der Basis-Klasse vorhanden, und wird versucht eine solche Klasse zu deserialisieren, so wird die Exception „java.io.InvalidClassException“ mit dem Fehler „NoSuchMethodError“ ausgelöst. 25.6 Aufgaben 25.6.1 Aufgabe „Scribble 7“ Erweitern sie das „Scribble 6“ um die Funktionalität das Bild zu speichern und zu laden. Lösung siehe Kapitel 25.7. 25.7 Lag. zu Aufgabe „Scribble 7“ – Kap. 25.6.1 todo... © Detlef Wilkening 1997-2016 http://www.wilkening-online.de