Das Behandeln von Ausnahmen erfolgt mit Hilfe der try-catch-Anweisung:
try {
Anweisung;
...
} catch (Ausnahmetyp x) {
Anweisung;
...
}
Der try-Block enthält dabei eine oder mehrere Anweisungen, bei deren Ausführung ein Fehler des Typs Ausnahmetyp auftreten kann. In diesem Fall wird die normale Programmausführung unterbrochen, und der Programmablauf fährt mit der ersten Anweisung nach der catch-Klausel fort, die den passenden Ausnahmetyp deklariert hat. Hier kann nun Code untergebracht werden, der eine angemessene Reaktion auf den Fehler realisiert.
Wir wollen das folgende fehlerhafte Programm betrachten:
public class Example0901 {
public static void main(String[] args)
{
int i, base = 0;
for (base = 10; base >= 2; --base) {
i = Integer.parseInt("40",base);
System.out.println("40 base "+base+" = "+i);
}
}
}
Hier soll der String "40" aus verschiedenen Zahlensystemen in ein int konvertiert und als Dezimalzahl ausgegeben werden. Die dazu verwendete Methode parseInt überprüft, ob der übergebene String einen gültigen Zahlenwert zur angegebenen Basis darstellt. Ist dies nicht der Fall, löst sie eine Ausnahme des Typs NumberFormatException aus. Ohne weitere Maßnahmen stürzt das Programm dann beim Versuch, den String "40" als Zahl zur Basis 4 anzusehen, ab:
40 base 10 = 40
40 base 9 = 36
40 base 8 = 32
40 base 7 = 28
40 base 6 = 24
40 base 5 = 20
java.lang.NumberFormatException: 40
at java.lang.Integer.parseInt(Integer.java:139)
at Example0901.main(Example0901.java:22)
Das Programm läßt sich nun leicht gegen solche Fehler absichern, indem die Programmsequenz, die den Fehler verursacht, in eine try-catch-Anweisung eingeschlossen wird:
public class Example0902 {
public static void main(String[] args)
{
int i, base = 0;
try {
for (base = 10; base >= 2; --base) {
i = Integer.parseInt("40",base);
System.out.println("40 base "+base+" = "+i);
}
} catch (NumberFormatException e) {
System.out.println("40 ist keine Zahl zur Basis "+base);
}
}
}
Zwar ist 40 immer noch keine Zahl zur Basis 4, aber das Programm fängt diesen Fehler nun ab und gibt eine geeignete Fehlermeldung aus:
40 base 10 = 40 40 base 9 = 36 40 base 8 = 32 40 base 7 = 28 40 base 6 = 24 40 base 5 = 20 40 ist keine Zahl zur Basis 4
In der catch-Klausel wird nicht nur die Art des abzufangenden Fehlers definiert, sondern auch ein formaler Parameter angegeben, der beim Auftreten der Ausnahme ein Fehlerobjekt übernehmen soll. Fehlerobjekte sind dabei Instanzen der Klasse Throwable oder einer ihrer Unterklassen. Sie werden vom Aufrufer der Ausnahme erzeugt und als Parameter an die catch-Klausel übergeben. Das Fehlerobjekt enthält Informationen über die Art des aufgetretenen Fehlers. So liefert beispielsweise die Methode getMessage einen Fehlertext, und printStackTrace druckt einen Auszug aus dem Laufzeit-Stack:
public String getMessage(); public void printStackTrace()
Unser Beispielprogramm könnte dann wie folgt umgeschrieben werden:
public class Example0903 {
public static void main(String[] args)
{
int i, base = 0;
try {
for (base = 10; base >= 2; --base) {
i = Integer.parseInt("40",base);
System.out.println("40 base "+base+" = "+i);
}
} catch (NumberFormatException e) {
System.out.println("***Fehler aufgetreten***");
System.out.println("Ursache: "+e.getMessage());
e.printStackTrace();
}
}
}
Die Ausgabe des Programms wäre dann:
40 base 10 = 40
40 base 9 = 36
40 base 8 = 32
40 base 7 = 28
40 base 6 = 24
40 base 5 = 20
***Fehler aufgetreten***
Ursache: 40
java.lang.NumberFormatException: 40
at java.lang.Integer.parseInt(Integer.java:229)
at Example0903.main(Example0903.java:8)
Wie man sieht, ähnelt die Ausgabe des Programms der ersten Version, die ohne expliziten Fehler-Handler geschrieben wurde. Das liegt daran, daß das Java-Laufzeitsystem bei Auftreten eines Fehlers, der von keiner Methode behandelt wurde, printStackTrace aufruft, bevor es das Programm beendet.
Alle Laufzeitfehler in Java sind Unterklassen der Klasse Throwable. Throwable ist eine allgemeine Fehlerklasse, die im wesentlichen eine Klartext-Fehlermeldung speichern und einen Auszug des Laufzeit-Stacks ausgeben kann. Unterhalb von Throwable befinden sich zwei große Vererbungshierarchien:
Viele Pakete der Java-Klassenbibliothek definieren ihre eigenen Fehlerklassen. So gibt es spezielle Fehlerklassen für die Dateiein- und -ausgabe, die Netzwerkkommunikation oder den Zugriff auf Arrays. Wir werden diese speziellen Fehlerklassen immer dann erläutern, wenn eine Methode besprochen wird, die diese Art von Fehler erzeugt.
Die Reaktion auf eine Ausnahme muß keinesfalls zwangsläufig darin bestehen, das Programm zu beenden. Statt dessen kann auch versucht werden, den Fehler zu beheben oder zu umgehen, um dann mit dem Programm fortzufahren. Wird im obigen Programm die try-catch-Anweisung in die Schleife gesetzt, so fährt das Programm nach jedem Fehler fort und versucht, die Konvertierung zur nächsten Basis vorzunehmen:
Beispiel
public class Example0904 {
public static void main(String[] args)
{
int i, base = 0;
for (base = 10; base >= 2; --base) {
try {
i = Integer.parseInt("40",base);
System.out.println("40 base "+base+" = "+i);
} catch (NumberFormatException e) {
System.out.println("40 ist keine Zahl zur Basis "+base);
}
}
}
}
Die Ausgabe des Programms lautet nun:
40 base 10 = 40 40 base 9 = 36 40 base 8 = 32 40 base 7 = 28 40 base 6 = 24 40 base 5 = 20 40 ist keine Zahl zur Basis 4 40 ist keine Zahl zur Basis 3 40 ist keine Zahl zur Basis 2
Bisher sind wir davon ausgegangen, daß innerhalb eines try-Blocks nur eine Ausnahme auftreten kann. Tatsächlich ist es natürlich ohne weiteres möglich, daß zwei oder mehrere unterschiedliche Ausnahmen ausgelöst werden. Das Programm kann auf verschiedene Fehler reagieren, indem es mehr als eine catch-Klausel verwendet. Jede catch-Klausel fängt die Fehler ab, die zum Typ des angegebenen Fehlerobjekts zuweisungskompatibel sind. Dazu gehören alle Fehler der angegebenen Klasse und all ihrer Unterklassen (das wichtige Konzept der Zuweisungskompatibilität von Objekttypen wurde in Kapitel 7 erläutert). Die einzelnen catch-Klauseln werden in der Reihenfolge ihres Auftretens abgearbeitet.
Wir wollen das Beispielprogramm nun so erweitern, daß nicht nur ein einzelner String in eine Zahl konvertiert wird, sondern ein Array von Strings. Aufgrund eines Fehlers in der verwendeten for-Schleife verursacht das Programm einen Zugriff auf ein nicht vorhandenes Array-Element und löst damit eine Ausnahme des Typs IndexOutOfBoundsException aus. Diese kann zusammen mit der Ausnahme NumberFormatException in einer gemeinsamen try-catch-Anweisung behandelt werden:
public class Example0905 {
public static void main(String[] args)
{
int i, j, base = 0;
String numbers[] = new String[3];
numbers[0] = "10";
numbers[1] = "20";
numbers[2] = "30";
try {
for (base = 10; base >= 2; --base) {
for (j = 0; j <= 3; ++j) {
i = Integer.parseInt(numbers[j],base);
System.out.println(numbers[j]+" base "+base+" = "+i);
}
}
} catch (IndexOutOfBoundsException e1) {
System.out.println(
"***IndexOutOfBoundsException: "+e1.getMessage()
);
} catch (NumberFormatException e2) {
System.out.println(
"***NumberFormatException: "+e2.getMessage()
);
}
}
}
Die Ausgabe des Programms ist nun:
10 base 10 = 10 20 base 10 = 20 30 base 10 = 30 ***IndexOutOfBoundsException: 3
Die try-catch-Anweisung enthält einen optionalen Bestandteil, der bisher noch nicht erläutert wurde. Mit Hilfe der finally-Klausel, die als letzter Bestandteil einer try-catch-Anweisung verwendet werden darf, kann ein Programmfragment definiert werden, das immer dann ausgeführt wird, wenn die zugehörige try-Klausel betreten wurde. Dabei spielt es keine Rolle, welches Ereignis dafür verantwortlich war, daß die try-Klausel verlassen wurde. Die finally-Klausel wird insbesondere dann ausgeführt, wenn der try-Block durch eine der folgenden Anweisungen verlassen wurde:
|
|
Die finally-Klausel ist also der ideale Ort, um Aufräumarbeiten durchzuführen. Hier können beispielsweise Dateien geschlossen oder Ressourcen freigegeben werden. |
Die folgende Variation unseres Beispielprogramms benutzt finally dazu, am Ende des Programms eine Meldung auszugeben:
public class Example0906 {
public static void main(String[] args)
{
int i, base = 0;
try {
for (base = 10; base >= 2; --base) {
i = Integer.parseInt("40",base);
System.out.println("40 base "+base+" = "+i);
}
} catch (NumberFormatException e) {
System.out.println("40 ist keine Zahl zur Basis "+base);
} finally {
System.out.println(
"Sie haben ein einfaches Beispiel " +
"sehr glücklich gemacht."
);
}
}
}
Die Ausgabe des Programms ist:
40 base 10 = 40 40 base 9 = 36 40 base 8 = 32 40 base 7 = 28 40 base 6 = 24 40 base 5 = 20 40 ist keine Zahl zur Basis 4 Sie haben ein einfaches Beispiel sehr glücklich gemacht.