ich war hier: Baumelement2871 » Baumelement5889 » Baumelement3368 » ObjProg03Exceptions
 (image: http://wdb.fh-sm.de/uploads/QualipaktLehre/BMBF_Logo_klein.jpg)

Objektorientierte Programmierung - Kapitel 3 - Exceptions


Inhalte von Dr. E. Nadobnyh

3.1. Traditionelle Fehlerbehandlung


Mögliche Fehlerursachen


1) Division durch Null,

2) zu große oder zu kleine Werte für einen Datentyp,

3) kein dynamischer Speicher mehr verfügbar,

4) Fehler beim Dateizugriff, z.B. Datei nicht vorhanden,

5) ungültige Adresse im Hauptspeicher,

6) fehlerhafte Eingaben durch den Benutzer,

7) Bereichsüberschreitung eines Arrays.


Fehlersituation


Eine Fehlersituation ist ein Verstoß gegen die Spezifikation der Operation. Wenn die Aufgabe der Operation nicht erfüllt ist, ist es meist sinnlos, weitere Anweisungen auszuführen.

Wegen der aufgetretenen Fehlersituation muss die normale Ausführungsreihenfolge unterbrochen werden, um die Fehlersituation zu behandeln.

Beispiel:

   if(b==0)
   {  //Die Fehlersituation ist ermittelt.
       //Die normale Ausführungsreihenfolge muss  
       //unterbrochen werden.
       //Ohne Behandlung tritt der Fehler auf.
   }
   c = a / b ;  



Erkennung und Behandlung von Fehler


1) Der Autor einer Bibliothek kann Fehlersituation erkennen, aber er weiß nicht, wie er sie behandeln soll.

Die Umgebung der Fehlersituation ist eine Stelle im Programm, wo die Fehlersituation entsteht.

2) Der Anwender einer Bibliothek weiß, wie er die Fehlersituation behandeln kann, hat aber keine Möglichkeit sie
zu entdecken.

Die Aufrufumgebung ist eine Stelle im Programm, wo die Fehlersituation behandelt werden kann.

Begriffsdiskussion: Fehlersituationsbehandlung und Fehlerbehandlung


Traditionelle Fehlerbehandlung. Fehlernummer


1) Meldungsausgabe und Programmabbruch, z.B. exit(-2);

2) Übergabe der Fehlernummer an den Aufrufer.

Die Nummer gibt Auskunft über Erfolg oder Misserfolg der Funktionsausführung und kann

  • als Parameter,
  • als Rückgabewert oder
  • mittels globaler Variable

zurückgeliefert werden.

3) Fehlerbehebung, z.B. durch Wiederholung der Eingabe

Demo 1. Fehlernummer


Traditionelle Behandlung. Nachteile


1) Jeder Programmierer entwickelt eigene Regeln für den Umgang mit Fehlersituationen.

Ein Beispiel ist die C-Funktion getchar(), die ein Zeichen liest und zurückliefert. Damit neben jedem beliebigen Zeichen auch ein Fehlerwert zurückgeliefert werden kann, ist der Typ des Rückgabewertes nicht char, sondern int.

2) Nach jedem Aufruf müssen die Fehlerabfragen durchgeführt werden. Dadurch wird das Programm beträchtlich aufgeblasen.


3.2. Grundbegriffe


Exception-Konzept


Eine Exception (Ausnahme) ist eine Kontrollstruktur, welche den Übergang (Sprung) von der Umgebung der Fehlersituation zur Aufrufumgebung erfüllt.

Vorteile:

1) Konzept beruht auf der Trennung des normalen Programmablaufs und der Behandlung von Fehlersituationen.

2) Fehlersituationen werden in der Aufrufumgebung behandelt. Die Fehlersituationen können zentral behandelt werden.

3) Das Programmverhalten wird überschaubar und sicherer.

Anmerkung: Durch eine Exception kann auch eine Fehlersituation bezeichnet werden.


Schlüsselwörter try, catch, throw


Die Ausnahmebehandlung lässt sich wie folgt skizzieren:

1) Eine Funktion versucht (try) die Erledigung einer Aufgabe.

2) Wenn diese Funktion eine Fehlersituation feststellt, die sie nicht beheben kann, wird ein Ausnahmeobjekt (Exception-Objekt, Fehlerobjekt) ausgeworfen (throw) bzw. wird eine Ausnahme (Exception) ausgelöst.

3) Dieses Ausnahmeobjekt wird von einer Ausnahmebehandlungsroutine (Exception-Handler) aufgefangen (catch), die die Fehlersituation bearbeitet.


Rückzugsvarianten


 (image: https://ife.erdaxo.de/uploads/ObjProg03Exceptions/oop13.gif)


Auslösen und Auffangen von Exceptions


Die throw- Anweisung wird in folgenden Schritten ausgeführt:

1) Fehlerobjekt auswerfen
Vom Parameter der throw- Anweisung wird eine Kopie gemacht und im Heap gespeichert.

2) Stack- Abwicklung
Die Veränderungen des Stacks, die seit dem Eintritt in den try-Block stattgefunden haben, werden rückgängig gemacht. Für alle zerstörten Objekte werden Destruktoren aufgerufen.

Vorsicht: throw ist kein goto.

3) Rückzug aus einer Fehlersituation
Ein passender catch-Block wird gefunden und ausgeführt. Das Fehlerobjekt wird als Argument übergeben.

4) Fehlerobjekt löschen
Beim Verlassen des catch- Blocks wird das Fehlerobjekt im Heap gelöscht.

5) Fortsetzung des Programms

⇒ Demo 2. Stack-Abwicklung


Registrierung von Exceptions


Jede Exception wird von einem Handler übernommen, wenn sie vorher in einem try-Block registriert wird.

Direkt bei Eintritt in ihren Handler wird eine Exception als behandelt betrachtet.

Beispiel:

Dieses Programm erzeugt keine Endlosschleife:

     void f()
     {    try {  throw ( 5 );}
           catch(int ) {  throw ( 6 );  }
     }  



3.3. Details


Unterscheidung von Exceptions


Man kann unterschiedliche Fehlersituationen durch unterschiedliche Werte des Fehlerobjektes darstellen.

Beispiel

 try
    {    ...;    if(dies) throw(7);    
          ...;    if(jenes) throw(12);    
          ...;    if(sonstwas) throw(25);  
    }
    catch(int  i)
    {      switch(i)
            {    case   7:  ...; break;  //dies    
                  case 12:  ...; break;  //jenes
                  case 25:  ...; break;  //sonstwas
            }
    }



Fehlerklassen


Normaleweise werden für verschiedene Fehler eigene Fehlerklassen definiert. Ein Objekt von Fehlerklasse kann genauere Informationen über die Fehlerursache enthalten.

Beispiel

class Error
  {    //Infos über Fehlerursache
  };
  void f1()
  {        
         if( ?? )
         {    Error  e1;
               //Infos über Fehlerursache in e1 speichern
               throw (e1);
         }
  }  



Handler-Suche


1) Jedem try-Block können mehrere catch- Blöcke (Exception-Handlers) zugeordnet werden.

2) Beim Rückzug aus Fehlersituation wird ein passender Handler gesucht, und zwar sequentiell, beginnend mit dem ersten catch-Block.

3) Es wird der erste Handler ausgeführt, dem das ausgeworfene Fehlerobjekt wie bei einem gewöhnlichen Aufruf übergeben werden kann (exception matching).

Vorsicht: throw ist kein Aufruf.


Genereller Exception-Handler


Als letzter catch- Block kann ein so genannter genereller Exception-Handler definiert werden.

Er übernimmt alle ausgelösten Exceptions, welche von keinem vorherigen Block übernommen werden.

   catch(…)
   {    //Handler für jede Art von Exceptoin
         //drei Punkte statt Typ
   }


Kann im Programm kein Handler gefunden werden, so wird das Programm beendet.

⇒ Demo 3. Handler-Suche
⇒ Demo 4. Erfolglose Handler-Suche


Anordnung von try-Blöcken


In einem Programm können mehrere try-Blöcke mit entsprechenden Exception-Handlern eingesetzt werden. Die Blöcke können sequentiell oder verschachtelt angeordnet werden.

Handler-Suche in verschachtelten Blöcken

1) Einige Exceptions können durch die Handler eines inneren try-Blocks separat behandelt werden.
2) Alle übrigen Exceptions suchen nach dem passenden Handler des äußeren try-Blocks.


Wiederauswerfen von Exception


Eine im inneren try-Block ausgelöste Exception kann an den umgebenen try- Block weitergereicht werden.

Die aktuell bearbeitete Exception wird nochmals ausgeworfen, um vom Handler des umgebenden Blocks
weiter bearbeitet zu werden.

 (image: https://ife.erdaxo.de/uploads/ObjProg03Exceptions/oop14.gif)


Exception- Spezifikation


Die Exceptions, die von einer Funktion ausgelöst werden kann, gehören zu den Eigenschaften einer Funktion. Sie müssen dem Anwender bekannt sein, damit geeignete Reaktionen programmiert werden können.

An die Funktions-Deklaration (Prototyp) wird einer so genannten Exception- Spezifikation angehängt.

Deklaration:
float pf(float y, float x) throw (DiviNull, Unterlauf);

Definition:
float pf(float y, float x) throw (DiviNull, Unterlauf){...}


Ist die Exception-Liste leer, so wird keine Exception zugelassen.

Deklaration:
float pf(float y, float x) throw ( );

Definition:
float pf(float y, float x) throw ( ){...}

Ist keine Exception- Spezifikation angegeben, so sind alle Exceptions zulässig.


Standard-Fehlerklassen


In C++ wird eine Reihe von Standard-Fehlerklassen definiert.

Einige Klassen:

Klassenname Geworfen von der Methode Bedeutung
bad_alloc new Speicherzuweisungsfehler
bad_cast dynamic_cast Typumwandlungsfehler
bad_typeid typeid Falscher Objekttyp
out_of_range at() Bereichsüberschreitung in Funktionen der C++ Standard-Bibliothek



⇒ Demo 5. Fehlerklasse bad_alloc




CategoryObjektorientierteProgrammierung
Diese Seite wurde noch nicht kommentiert.
Valid XHTML :: Valid CSS: :: Powered by WikkaWiki