Galileo Computing < openbook > Galileo Computing - Professionelle Bücher. Auch für Einsteiger.
Professionelle Bücher. Auch für Einsteiger.

Inhaltsverzeichnis
Vorwort
1 Java ist auch eine Sprache
2 Sprachbeschreibung
3 Klassen und Objekte
4 Der Umgang mit Zeichenketten
5 Mathematisches
6 Eigene Klassen schreiben
7 Angewandte Objektorientierung
8 Exceptions
9 Die Funktionsbibliothek
10 Threads und nebenläufige Programmierung
11 Raum und Zeit
12 Datenstrukturen und Algorithmen
13 Dateien und Datenströme
14 Die eXtensible Markup Language (XML)
15 Grafische Oberflächen mit Swing
16 Grafikprogrammierung
17 Netzwerkprogrammierung
18 Verteilte Programmierung mit RMI und Web-Services
19 JavaServer Pages und Servlets
20 Applets
21 Midlets und die Java ME
22 Datenbankmanagement mit JDBC
23 Reflection und Annotationen
24 Logging und Monitoring
25 Sicherheitskonzepte
26 Java Native Interface (JNI)
27 Dienstprogramme für die Java-Umgebung
A Die Begleit-DVD
Stichwort

Download:
- ZIP, ca. 12,5 MB
Buch bestellen
Ihre Meinung?

Spacer
<< zurück
Java ist auch eine Insel von Christian Ullenboom
Programmieren mit der Java Standard Edition Version 6
Buch: Java ist auch eine Insel

Java ist auch eine Insel
7., aktualisierte Auflage
geb., mit DVD (November 2007)
1.492 S., 49,90 Euro
Galileo Computing
ISBN 978-3-8362-1146-8
Pfeil 8 Exceptions
Pfeil 8.1 Problembereiche einzäunen
Pfeil 8.1.1 Exceptions in Java mit try und catch
Pfeil 8.1.2 Eine Datei mit RandomAccessFile auslesen
Pfeil 8.1.3 Ablauf einer Ausnahmesituation
Pfeil 8.1.4 Wiederholung abgebrochener Bereiche
Pfeil 8.1.5 throws im Methodenkopf angeben
Pfeil 8.1.6 Abschlussbehandlung mit finally
Pfeil 8.1.7 Nicht erreichbare catch-Klauseln
Pfeil 8.2 Die Klassenhierarchie der Fehler
Pfeil 8.2.1 Die Exception-Hierarchie
Pfeil 8.2.2 Oberausnahmen auffangen
Pfeil 8.2.3 Alles geht als Exception durch
Pfeil 8.2.4 RuntimeException muss nicht aufgefangen werden
Pfeil 8.2.5 Harte Fehler: Error
Pfeil 8.3 Auslösen eigener Exceptions
Pfeil 8.3.1 Mit throw Ausnahmen auslösen
Pfeil 8.3.2 Neue Exception-Klassen deklarieren
Pfeil 8.3.3 Abfangen und Weiterleiten
Pfeil 8.3.4 Geschachtelte Ausnahmen
Pfeil 8.3.5 Rückgabewerte bei ausgelösten Ausnahmen
Pfeil 8.4 Der Stack-Trace
Pfeil 8.4.1 Stack-Trace erfragen
Pfeil 8.5 Assertions
Pfeil 8.5.1 Assertions in eigenen Programmen nutzen
Pfeil 8.5.2 Assertions aktivieren


Galileo Computing - Zum Seitenanfang

8.3 Auslösen eigener Exceptions Zur nächsten ÜberschriftZur vorigen Überschrift

Bisher wurden Exceptions lediglich aufgefangen, aber noch nicht selbst erzeugt. In diesem Kapitel wollen wir sehen, wie eigene Ausnahmen ausgelöst werden. Das kann zum einen erfolgen, wenn die JVM provoziert wird, etwa bei einer ganzzahligen Division durch 0, oder explizit durch throw.


Galileo Computing - Zum Seitenanfang

8.3.1 Mit throw Ausnahmen auslösen Zur nächsten ÜberschriftZur vorigen Überschrift

Soll eine Funktion selbst eine Exception auslösen, muss sie zunächst ein Exception-Objekt erzeugen und dann die Ausnahmebehandlung anstoßen. Im Sprachschatz dient das Schlüsselwort throw dazu, eine Ausnahme auszulösen. Als Exception-Typ soll IllegalArgumentException dienen, das ein fehlerhaftes Argument anzeigt.

Listing 8.11 com/tutego/insel/exceptions/v1/Player.java, Konstruktor

Player( int age ) 
{ 
  if ( age <= 0 ) 
    throw new IllegalArgumentException( "No age <= 0 allowed!" ); 
 
  this.age = age; 
}

Wir sehen im Beispiel, dass negative oder null Alter-Übergaben nicht gestattet sind und zu einem Fehler führen. Im ersten Schritt baut dazu der new-Operator das Exception-Objekt über einen parametrisierten Konstruktor auf. Die Klasse IllegalArgumentException bietet einen solchen Konstruktor, der eine Zeichenkette annimmt, die den näheren Grund der Ausnahme übermittelt. Welche Parameter die einzelnen Exception-Klassen deklarieren, ist der API zu entnehmen. Nach dem Aufbau des Exception-Objekts beendet throw die lokale Abarbeitung, und die JVM sucht ein catch, das die Ausnahme behandelt.

Lassen wir ein Beispiel folgen, in dem Spieler mit einem negativen Alter initialisiert werden sollen:

Listing 8.12 com/tutego/insel/exceptions/v1/Player.java, main()

try 
{ 
  Player d = new Player( –100 ); 
  System.out.println( d ); 
} 
catch ( IllegalArgumentException e ) 
{ 
  e.printStackTrace(); 
}

Das führt zu einer Exception, und der Stack-Trace, der den printStackTrace() ausgibt, ist:

java.lang.IllegalArgumentException: No age <= 0 allowed! 
    at com.tutego.insel.exceptions.v1.Player.<init>(Player.java:10) 
    at com.tutego.insel.exceptions.v1.Player.main(Player.java:19)

Hinweis Hinweis Löst ein Konstruktor eine Ausnahme aus, ist eine Nutzung wie die folgende problematisch:

Player p = null; 
try { p = new Player(v); } catch ( IllegalArgumentException e ) { } 
p.getAge();   // (*) BUM!

Die Exception führt zu keinem Player-Objekt, wenn v negativ ist. So bleibt p mit null vorbelegt. Es folgt in der BUM-Zeile eine NullPointerException. Der Programmcode, der das Objekt erwartet, sollte also mit in den try-catch-Block.


Da die IllegalArgumentException eine RuntimeException ist, hätte es in main() auch ohne try-catch heißen können:

public static void main( String[] args ) 
{ 
  Player d = new Player( –100 ); 
}

Die Runtime-Exception müsste nicht zwingend aufgefangen werden, aber der Effekt wäre, dass die Ausnahme nicht behandelt würde und das Programm abbricht.


class java.lang.IllegalArgumentException 
extends RuntimeException

  • IllegalArgumentException() Erzeugt eine neue Ausnahme ohne genauere Fehlerangabe.
  • IllegalArgumentException( String s ) Erzeugt ein neues Fehler-Objekt mit einer detaillierteren Fehlerangabe.

Galileo Computing - Zum Seitenanfang

8.3.2 Neue Exception-Klassen deklarieren Zur nächsten ÜberschriftZur vorigen Überschrift

Eigene Exceptions sind immer direkte (oder indirekte) Unterklassen von Exception. Sie implementieren oft zwei Konstruktoren: einen Standard-Konstruktor und einen mit einem String parametrisierten Konstruktor. Um für die Klasse Player im letzten Beispiel einen neuen Fehlertyp zu deklarieren, erweitern wir RuntimeException zur PlayerException.

Listing 8.13 com/tutego/insel/exception/v2/PlayerException.java

package com.tutego.insel.exception.v2; 
 
public class PlayerException extends RuntimeException 
{ 
  public PlayerException() 
  { 
  } 
 
  public PlayerException( String s ) 
  { 
    super( s ); 
  } 
}

Nehmen wir uns die Initialisierung mit dem Alter noch einmal vor. Statt der IllegalArgument-Exception löst der Konstruktor im Fehlerfall unsere speziellere PlayerException aus.

Listing 8.14 com/tutego/insel/exception/v2/Player.java, Ausschnitt

if ( age <= 0 ) 
  throw new PlayerException( "No age <= 0 allowed!" );

Im Hauptprogramm können wir auf die PlayerException reagieren, indem wir die Ausnahme explizit mit try-catch auffangen oder an den Aufrufer weitergeben – unsere Exception ist ja eine RuntimeException und müsste nicht direkt abgefangen werden.

Exception in thread "main" com.tutego.insel.exceptions.v2.PlayerException:  
No age <= 0 allowed! 
    at com.tutego.insel.exceptions.v2.Player.<init>(Player.java:10) 
    at com.tutego.insel.exceptions.v2.Player.main(Player.java:19)

Tipp Tipp Es ist immer eine gute Idee, Unterklassen von Exception zu bauen. Würden wir keine Unterklassen anlegen, sondern direkt mit throw new Exception() einen Fehler anzeigen, so könnten wir unseren Fehler später nicht mehr von anderen Fehlern unterscheiden. Mit der Hierarchiebildung wird nämlich die Spezialisierung bei mehreren catch-Anweisungen sowie eine Unterscheidung mit instanceof unterstützt. Wir müssen immer unseren Fehler mit catch(Exception e) auffangen und bekommen so alle anderen Fehler mit aufgefangen, die dann nicht mehr unterschieden werden können. Allerdings sollten Entwickler nicht zu inflationär mit den Ausnahmen-Hierarchien umgehen; in vielen Fällen reicht eine Standard-Ausnahme aus.



Galileo Computing - Zum Seitenanfang

8.3.3 Abfangen und Weiterleiten Zur nächsten ÜberschriftZur vorigen Überschrift

Die Ausnahme, die ein try-catch-Block auffängt, kann in catch wieder neu ausgelöst werden. Dieses Re-throwing zeigt folgendes Beispiel, in dem das Programm den Fehler erst über einen Logger ausgibt und dann weiter nach oben reicht.

Listing 8.15 Rethrow.java

import java.util.logging.Logger; 
 
public class Rethrow 
{ 
  private static Logger log = Logger.getAnonymousLogger(); 
 
  static void rethrow( RuntimeException e ) 
  { 
    log.warning( "RuntimeException occurred!" ); 
 
//    e.fillInStackTrace(); 
 
    throw e; 
  } 
 
  public static void main( String[] args ) 
  { 
    try 
    { 
      ((String)null).length(); 
    } 
    catch ( NullPointerException e ) 
    { 
      rethrow( e ); 
    } 
  } 
}

Die Ausgabe ist:

16.08.2006 18:35:27 Rethrow rethrow 
WARNUNG: RuntimeException occurred! 
Exception in thread "main" java.lang.NullPointerException 
    at Rethrow.main(Rethrow.java:20)

Wichtig an dieser Stelle ist die Tatsache, dass der Aufrufstack in e gespeichert ist und daher der Strack-Trace nicht die Funktion rethrow() enthält. Wünschten wir dies, müssen wir den Strack-Trace neu füllen. Dazu dient die Funktion fillInStackTrace(). Nehmen wir den Aufruf im oberen Beispiel hinein, folgt die Ausgabe:

16.08.2006 18:36:16 Rethrow rethrow 
WARNUNG: RuntimeException occurred! 
Exception in thread "main" java.lang.NullPointerException 
    at Rethrow.rethrow(Rethrow.java:11) 
    at Rethrow.main(Rethrow.java:24)

Galileo Computing - Zum Seitenanfang

8.3.4 Geschachtelte Ausnahmen Zur nächsten ÜberschriftZur vorigen Überschrift

Besonders bei eigenen Ausnahmen mag ein Grund für die Auslösung sein, dass ein eingebetteter Teil versagt. Das ist vergleichbar mit einer Transaktion: Ist ein Teil der Kette fehlerhaft, so ist der ganze Teil nicht ausführbar. Bei unseren Ausnahmen ist das nicht anders. Nehmen wir an, wir haben eine Methode, die im Falle eines Misslingens eine Ausnahme auslöst. Ruft unsere Methode nun eine Unterfunktion auf, die zum Beispiel eine Ein-/Ausgabeoperation tätigt, was jedoch schiefgeht, wird der Anlass für unsere Exception die IO-Exception sein. Es liegt also nahe, bei der Nennung des Grundes für das eigene Versagen die Unteraufgabe zu nennen. (Wieder ein Beweis, wie »menschlich« Programmieren sein kann.)

Eine geschachtelte Ausnahme (engl. nested exception) speichert einen Verweis auf eine weitere Ausnahme. Wenn ein Exception-Objekt aufgebaut wird, lässt sich der Grund (engl. cause) als Argument im Konstruktor der Throwable-Klasse übergeben. Die Ausnahme-Basisklasse bietet dafür zwei Konstruktoren:

  • Throwable( Throwable cause )
  • Throwable( String message, Throwable cause )

Der Grund kann über die Methode Throwable getCause() erfragt werden.

Da Konstruktoren in Java nicht vererbt werden, bieten die Unterklassen oft Konstruktoren an, um den Grund anzunehmen: Zumindest Exception macht das und kommt somit auf vier Erzeuger:

  • Exception()
  • Exception( String message )
  • Exception( String message, Throwable cause )
  • Exception( Throwable cause )

Einige der tiefer liegenden Unterklassen haben dann auch diese Konstruktor-Typen mit Throw-able-Parameter, wie IOException, SQLException oder ClassNotFoundException, andere wiederum nicht, wie PrinterException. Eigene Unterklassen können auch mit initCause() genau einmal eine geschachtelte Ausnahme angeben.

In modernen Frameworks ist die Nutzung von Ausnahmen, die nicht geprüft werden müssen, also Exemplare von RuntimeException sind, häufiger geworden. Bekannte zu prüfende Ausnahmen werden in RuntimeException-Objekte verpackt (eine Art Exception-Wrapper), die den Verweis auf die auslösende Nicht-RuntimeException speichern.


Galileo Computing - Zum Seitenanfang

8.3.5 Rückgabewerte bei ausgelösten Ausnahmen topZur vorigen Überschrift

Java versucht, durch den Programmfluss den Ablauf innerhalb einer Methode zu bestimmen und zu melden, ob sie definitiv einen Rückgabewert liefert. Dabei verfolgt der Compiler die Programmpfade und wertet bestimmte Ausdrücke aus. Doch die Aussage »Jede Funktion mit Ergebnistyp ungleich void muss eine return-Anweisung besitzen« müssen wir etwas relativieren. Nur in einem speziellen Fall müssen wir dies nicht. Nämlich genau dann, wenn vor dem Ende der Funktion eine throws-Anweisung die Abarbeitung beendet.

Listing 8.16 com/tutego/insel/exceptions/NoReturn.java

package com.tutego.insel.exception; 
 
class WindowsVistaKeyGenerator 
{ 
  public String generateKey() 
  { 
    throw new UnsupportedOperationException(); 
  } 
}

Ein Blick auf generateKey() verrät, dass trotz angekündigtem Rückgabewert keine return-Anweisung im Rumpf steht. Die Abarbeitung wird vor dem Rücksprung durch eine Exception abgebrochen. generateKey() muss diese Exception nicht mit throws ankündigen, da UnsupportedOperationException eine RuntimeException ist.



Ihr Kommentar

Wie hat Ihnen das <openbook> gefallen? Wir freuen uns immer über Ihre freundlichen und kritischen Rückmeldungen.






<< zurück



Copyright © Galileo Press 2008
Für Ihren privaten Gebrauch dürfen Sie die Online-Version natürlich ausdrucken. Ansonsten unterliegt das <openbook> denselben Bestimmungen, wie die gebundene Ausgabe: Das Werk einschließlich aller seiner Teile ist urheberrechtlich geschützt. Alle Rechte vorbehalten einschließlich der Vervielfältigung, Übersetzung, Mikroverfilmung sowie Einspeicherung und Verarbeitung in elektronischen Systemen.


[Galileo Computing]

Galileo Press, Rheinwerkallee 4, 53227 Bonn, Tel.: 0228.42150.0, Fax 0228.42150.77, info@galileo-press.de