Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Netzwerke (https://www.delphipraxis.net/14-netzwerke/)
-   -   FreeAndNIL mit Nebenwirkung (https://www.delphipraxis.net/211949-freeandnil-mit-nebenwirkung.html)

Maekkelrajter 25. Nov 2022 11:34

FreeAndNIL mit Nebenwirkung
 
Eine Routine zur Abfrage von Album-Daten von der Spotify Web API zeigt ein merkwürdiges Verhalten:
Delphi-Quellcode:
Function TSpManager.CCRequest(accesspoint:string; offset: Integer = 0; Limit: Integer = 50):TJSONValue;
  var
       Client: TRestClient;
       Request: TRESTRequest;
       URL: string;
begin
  result:= NIL;
  If not CCAuthenticationValid Then exit;
  URL:= 'https://api.spotify.com' + accesspoint;
  Client:= TRESTClient.Create(URL);
  Request:= TRESTRequest.Create(Client);
  try
    Client.contentType:= 'application/json';
    Client.Accept:= 'application/json';
    Request.AddParameter('Authorization','Bearer ' +  stcf.AccessData.CCAccessToken , pkHTTPHEADER, [poDoNotEncode]);
    Request.AddParameter('offset',offset.ToString,pkGetOrPost, [poDoNotEncode]);
    Request.AddParameter('limit',limit.ToString,pkGetOrPost, [poDoNotEncode]);
    Request.Execute;
    If Request.Response.StatusCode = 200 Then
    begin
      result:= Request.response.JSONValue; // im Debugger: TJSONObject(result) = (True, $3277D20) (korrekt)
    end;
  finally
    FreeAndNIL(Client);                    // im Debugger:TJSONObject(result) = (True, nil) (Fehler)
  end;
end;
Die Abfrage ist OK und liefert das gewünschte Ergebnis. Aber auf geheimnisvolle Weise zerstört das 'FreeAndNIL' den Rückgabewert und erzeugt bei dessen weiterer Vearbeitung Accessviolations.
Mache ich irgendwas falsch oder ist das womöglich ein Bug? Vielleicht weiß hier jemand mehr.
Als Workaround erzeuge ich Request und Client im 'OnCreate' der TSpManager - Klasse und entsorge sie im OnDestroy; Das funktioniert einwandfrei.

Gruß LP

Neutral General 25. Nov 2022 11:40

AW: FreeAndNIL mit Nebenwirkung
 
Du kannst auch eine Kopie der Response-Daten zurückgeben statt einer Referenz.
Dann sollte das auch funktionieren

DeddyH 25. Nov 2022 12:00

AW: FreeAndNIL mit Nebenwirkung
 
Der Request "gehört" dem Client und dem wiederum "gehört" der JSONValue. Gibt man also den Client frei (wieso man FreeAndNil auf lokale Variablen anwendet, wird mir immer ein Rätsel bleiben, aber egal), sind auch die anderen beiden weg. Es bleibt also nur, eine Kopie des JSONValues zu erzeugen und den dann zurückzugeben, wie Neutral General bereits erwähnt hat. Die muss man dann aber auch irgendwann wieder selbst freigeben.

Maekkelrajter 26. Nov 2022 21:58

AW: FreeAndNIL mit Nebenwirkung
 
Danke für die erhellenden Hinweise!
Ja, da fällt es mir wie Schuppen von den Augen, zumal ich nicht zum erstenmal mit dem Problem zu tun habe, dass Objekt-Variablen eben 'nur' Zeiger sind. :shock:
Ich nehme an, dass im vorliegenden Fall mit 'Kopie' eine sog. 'Deep Copy' gemeint ist. Das ist ja eigentlich ziemlich umständlich, aber es funktioniert:
Delphi-Quellcode:
function DeepCopy(aValue: TObject): TObject;
var
  MarshalObj: TJSONMarshal;
  UnMarshalObj: TJSONUnMarshal;
    JSONValue: TJSONValue;
begin
  Result:= nil;
  MarshalObj := TJSONMarshal.Create;
  UnMarshalObj := TJSONUnMarshal.Create;
  try
    JSONValue := MarshalObj.Marshal(aValue);
    try
      if Assigned(JSONValue) then
         Result:= UnMarshalObj.Unmarshal(JSONValue);
    finally
      JSONValue.Free;
    end;
  finally
     MarshalObj.Free;
     UnMarshalObj.Free;
  end;
end;

Function TSpManager.CCRequest(accesspoint:string; offset: Integer = 0; Limit: Integer = 50):TJSONValue;
    var Client: TRestClient;
        Request: TRESTRequest;
        URL: string;
begin
  result:= NIL;
  URL:= 'https://api.spotify.com' + accesspoint;
  Client:= TRESTClient.Create(URL);
  Request:= TRESTRequest.Create(Client);
  try
    [...]
     Request.Execute;
     If Request.Response.StatusCode = 200 Then
     begin
       result:= TJSONValue(DeepCopy(TObject(Request.response.JSONValue)));
     end;
  finally
    Client.free;
  end;
end;
Lästig dabei ist allerdings, dass die Kopie explizit freigegeben werden muss, was auch bei der folgenden Alternative nötig ist:
Delphi-Quellcode:
Function TSpManager.CCRequest(accesspoint:string; offset: Integer = 0; Limit: Integer = 50):TJSONValue;
    var Client: TRestClient;
        Request: TRESTRequest;
        URL: string;
begin
  [...]
  try
    [...]
    Request.Execute;
    If Request.Response.StatusCode = 200 Then
    begin
      result := TJSONObject.ParseJSONValue(TEncoding.utf8.GetBytes(Request.response.content), 0)
    end;
  finally
    Client.free;
  end;
end;
Letztlich habe ich das Problem gelöst, indem ich FRequest und FClient als Class Member deklariert habe.
Das funktioniert anscheinend zuverlässig und vermutlich schneller als die o.g. Alternativen

Gruß LP


Alle Zeitangaben in WEZ +1. Es ist jetzt 06:01 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