![]() |
Fehlerbehandlung so richtig
Hallo,
in die folgende (unübersichtliche) Funktion (mit Unterfunktionen) habe ich eine Fehlerbehandlung eingeführt. Ich weiß aber nicht, ob die funktioniert wie ich das denke (siehe Kommentare im Code), da bisher noch keine Fehler auftraten :-D. Meint ihr das klappt so, hab versucht was hier in der DP in dem Tutorial zu Exceptions gesagt wurde umzusetzen und was in der Hilfe so stand?
Delphi-Quellcode:
Hoffe diese Form der Fragestellung ist OK, so.
function TLToA.CreateDateien(i: Integer): Boolean;
var q : TADOQuery; ///////////////////////////// function SQL_Out : String; var s : TSQL; begin s:=TSQL.create; s.Add := 'Select Distinct exportline'; s.Add := 'From ('+SQL_Main(i)+')'; s.Add := 'Where gruppe = '+s.Ora_StrOfStr(q.FieldByName('gruppe').AsString); Result:=s.Text; s.Free; end; ///////////////////////////// procedure CreateGruppendatei; var Z : TADOQuery; T : TStringlist; Dateiname : String; begin T:=TStringList.Create; Dateiname:=q.FieldByName('datei').AsString; if Fileexists(Dateiname) then T.LoadFromFile(Dateiname); Z:=TADOQuery.Create(nil); Z.Connection:=aMain.Main.Projekt.DB.Con; Z.SQL.Text:=SQL_Out; try Z.Open; Z.First; while not Z.Eof do begin T.Add(Z.FieldByName('exportline').AsString); Z.Next; end; T.SaveToFile(Dateiname); except on e : Exception do begin // Innerer try-Block in Unterfunktion, soll reagieren auf Fehler // mit der Datenbank (Z.Open)und auf Fehler beim Speichern der TStringList (T.Save...) // Objekte der Unterfunktion sollen freigegeben werden T.Free; //Nötig? oder werden Freigaben 8 Zeilen tiefer trotzdem durchlaufen? Z.Free; // Dann soll der Fehler an die Hauptprozedur übergeben werden, weil da die // Meldung protokoliert werden soll. e.RaiseOuterException(e); //Exception weiterreichen? end; end; Z.Free; //Werden diese auch im Fehlerfall durchlaufen? T.Free; end; /////////////////////////// begin q := TADOQuery.Create(nil); q.Connection:=aMain.Main.Projekt.DB.Con; q.SQL.Add('Select'); q.SQL.Add(' Distinct gruppe, datei'); q.SQL.Add('From'); q.SQL.Add(' ('+SQL_Main(i)+')'); Result:=true; try q.Open; q.First; while not q.Eof do begin CreateGruppendatei; q.Next; end; except on e : Exception do begin // Hier sollen Fehler abgefangen werden, bei Problemen mit der DB (q.Open) // und die Fehler in evtl. aus der Unterfunktion hochgereicht werden // Z.Zt. ist das glaub ich so, dass die Schleife beendet wird (wg. dem q.free?) // wenn in der U-Funk ein Fehler passiert ist. Kann man das daher auch weglassen? // Würde dann der nächste Schleifendurchlauf gemacht oder ist die Schleife // nach einer Exception auf jeden Fall zu Ende? amain.Main.Projekt.MSG.Info('Fehler in Schritt '+IntToStr(i)).Text:=e.Message; Result:=false; q.Free; end; q.Free; end; end; |
AW: Fehlerbehandlung so richtig
Hey,
wenn es dir nur darum geht, dass der Speicher sauber freigegeben wird, würde ich dir empfehlen auf das try..finally Konstrukt zurückzugreifen:
Delphi-Quellcode:
function TLToA.CreateDateien(i: Integer): Boolean;
var q : TADOQuery; ///////////////////////////// function SQL_Out : String; var s : TSQL; begin s:=TSQL.create; try s.Add := 'Select Distinct exportline'; s.Add := 'From ('+SQL_Main(i)+')'; s.Add := 'Where gruppe = '+s.Ora_StrOfStr(q.FieldByName('gruppe').AsString); Result:=s.Text; finally s.Free; end; end; ///////////////////////////// // Ich arbeite sehr gerne auch mit with ... do function SQL_Out : String; begin with TSQL.create do try Add := 'Select Distinct exportline'; Add := 'From ('+SQL_Main(i)+')'; Add := 'Where gruppe = '+Ora_StrOfStr(q.FieldByName('gruppe').AsString); Result := Text; finally Free; end; end; |
AW: Fehlerbehandlung so richtig
Und man kann try-finally und try-except auch locker verschachteln.
Delphi-Quellcode:
Dings := TDings.Create;
try try Dings.MachWasGefährliches; except on E: Exception do begin Uiuiui; end; finally Dings.Free; end; |
AW: Fehlerbehandlung so richtig
Wo landet bei Verschachtelung den mein Raise? Wenn ich in der inneren "raise", landet es in der äußeren, oder?
Bau ich die verschachtelng also in die Unterfunktion, krieg ich es mit dem Raise ja nicht mehr ganz nach oben in die Hauptfunktion, wo ich das e.Message brauch... Oder gibts im finally auchnochmal e.Message und Raise? Meine Frage bezog sich somit weniger auf den einzelnen Block, als auf das Gesamtzusammenspiel |
AW: Fehlerbehandlung so richtig
Hi,
Wenn du eine Exception raise dann gelangt sie an die höherliegende "Ebene". Zitat:
Mfg Coffeecoder |
AW: Fehlerbehandlung so richtig
Im Except-Block machst Du irgendetwas und löst die Exception erneut aus. Das Finally sorgt ja nur dafür, dass die Objekte auch im Ausnahmefall wieder freigegeben werden, abgefangen wird da nichts.
Ein Beispiel aus meiner IPv4-Klasse:
Delphi-Quellcode:
procedure TCustomIPv4.SetAsString(const Value: string);
procedure IPError; begin raise Exception.CreateFmt(INVALID_IP, [Value]); end; procedure StrToByte(const s: string; var b: Byte); var dummy: integer; begin if not TryStrToInt(s, dummy) or (dummy > MAXBYTE) or (dummy < 0) then IPError else b := dummy; end; var sl: TStringlist; Rec: TIPAddressRec; begin sl := TStringlist.Create; try sl.Delimiter := '.'; sl.DelimitedText := StringReplace(Value, #32, '', [rfReplaceAll]); if sl.Count = 4 then try StrToByte(sl[0], Rec.ByteHiHi); StrToByte(sl[1], Rec.ByteHiLo); StrToByte(sl[2], Rec.ByteLoHi); StrToByte(sl[3], Rec.ByteLoLo); except raise; end else IPError; FIp := Rec.Mask; finally sl.Free; end; end; |
Alle Zeitangaben in WEZ +1. Es ist jetzt 18:24 Uhr. |
Powered by vBulletin® Copyright ©2000 - 2025, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024-2025 by Thomas Breitkreuz