Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Programmieren allgemein (https://www.delphipraxis.net/40-programmieren-allgemein/)
-   -   Fehlerbehandlung so richtig (https://www.delphipraxis.net/160266-fehlerbehandlung-so-richtig.html)

Jumpy 5. Mai 2011 10:36

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:
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;
Hoffe diese Form der Fragestellung ist OK, so.

geskill 5. Mai 2011 10:46

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;

DeddyH 5. Mai 2011 10:50

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;

Jumpy 5. Mai 2011 11:00

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

Coffeecoder 5. Mai 2011 11:04

AW: Fehlerbehandlung so richtig
 
Hi,

Wenn du eine Exception raise dann gelangt sie an die höherliegende "Ebene".
Zitat:

Wo landet bei Verschachtelung den mein Raise? Wenn ich in der inneren "raise", landet es in der äußeren, oder?
Sie landet in der Äußeren. Damit sie ganz nach oben wandert musst du die immer raise.

Mfg Coffeecoder

DeddyH 5. Mai 2011 11:14

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 16:42 Uhr.

Powered by vBulletin® Copyright ©2000 - 2024, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024 by Thomas Breitkreuz