Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Object-Pascal / Delphi-Language (https://www.delphipraxis.net/32-object-pascal-delphi-language/)
-   -   Delphi Via property auf Array zugreifen (Threadsicher?) (https://www.delphipraxis.net/109142-via-property-auf-array-zugreifen-threadsicher.html)

TheMiller 25. Feb 2008 08:55


Via property auf Array zugreifen (Threadsicher?)
 
Hallo,

nachdem ich große Hilfestellung bekommen habe, habe ich mir den mir genannten Vorschlag zu Herzen genommen, mich eingelesen und an meine Bedürfnisse weiter angepasst. Nun würde ich gerne wissen, ob das was ich gemacht habe, auch wirklich Threadsafe ist und wie ich das testen kann.

Was das Programm macht:

Das Hauptprogramm ruft eine DLL-Form auf, die per Windows.SetParent in die Hauptanwendung eingebunden wird. Danach läd die DLL-Form über eine Thread-Unit Daten, die im Thread bearbeitet werden. Die DLL-Form hat ein Container-Objekt erstellt und bei Treaderstellung an den Thread mit weitergegeben. Die Resultate der Bearbeiteten Daten werden in einem Array in der Container-Klasse gespeichert. Nach Beendigung sendet der Thread eine Message an das Hauptprogramm, welches die Daten aus dem ContainerArray ausliest und das Objekt wieder freigibt, wenn es nix mehr zu tun geben sollte.

Hier mal die entscheidenen Quelltexte:

Container:

Delphi-Quellcode:
type
  TExpData = Array of String;
type
  TContainer=class(TSimpleRWSync)
  private
    FExpData: TExpData;
   
    function getExpData(i: Integer): String;
    procedure setExpData(i: Integer; const Value: String);
    function getExpDataCount: Integer;
  public
    property ExpData[i: Integer]: String read getExpData write setExpData;
    property ExpDataCount: Integer read getExpDataCount;
  end;

function TContainer.getExpData(i: Integer): String;
begin
  BeginRead;
  result:=FExpData[i];
  EndRead;
end;

//Bei folgender Prozedur muss ich die Variable "i" haben, da sonst bei der property die Fehlermeldung "Inkompatible Typen" auftritt.
//Aber ich möchte das Array doch immer nur erhöhen.

procedure TContainer.setExpData(i: Integer; const Value: String);
begin
  BeginWrite;
  SetLength(FExpData, Succ(Length(FExpData)));
  FExpData[High(FExpData)]:=Value; <- Warum muss ich einen Index haben?
  EndWrite;
end;
Thread:

Execute ruft Methode zur Speicherung der Daten in ein Array auf. Das Array ist notwendig. Am Ende wird eine Message an die Hauptanwendung DLL-Form gesendet:

Delphi-Quellcode:
  SendMessage(FHndl, WM_CALLBACK, cb_id, LongInt(PChar(s)));
Somit weis die Form, dass alle Daten da sind und sie damit arbeiten kann.

Alle Methoden etc. arbeiten miteinander, funktionieren und die richten Daten kommen richtig dort an, wo sie ankommen sollen, doch meine Frage ist nun - da das noch relativ neu für mich war - ob das Threadsafe ist und wie ich das testen kann.

Wäre nett, wenn ihr mal drüberschauen könntent. (Achso, und ich habe im Quelltext noch eine Frage zur Indexvariablen. Vielleicht könnt ihr mir da noch einen Tipp geben.

Danke im Voraus!

sirius 25. Feb 2008 09:51

Re: Via property auf Array zugreifen (Threadsicher?)
 
Zum Index:
Anscheinend brauchst du den nicht, da du beim Schreiben immer nur hinten anhängen willst. Dann lass doch den Index im Property und in der set-Methode weg.
Den Index bräcuhtest du, wenn du irgendein Element des Arrays ändern willst. Aber du willst ja nur anhängen.
btw.: Warum benutzt du keine Stringliste. Wäre OOP-konformer, die Array-Lösung wie du sie derzeit umgesetzt hat, wird mit zunehmender Zahl an Elementen langsamer.

Zum Container:
Das hatten wir ja im anderen Thread soweit. Du hast aber getExpDataCount Threadsafe mit begin- und endread ausgeführt?
Und wie gesagt: Du hast TContainer jetzt von TSimpleRWSync abgeleitet. Du kannst ohne Probleme auch von TMultiReadExclusiveWriteSynchronizer (kurz: TMREWSync) ableiten. Dahinter steckt eine andere Art des Synchronisierens (je nach Anwendung ist das eine oder das andere schneller). Die öffentlichen Methoden sind netterweise dieselben. Dadurch muss man sonst nix ändern.


zum Callback:
Hier hast du jetzt eine zweite Variante des Synchronisierens hineingebracht. wenn du bei der schönen ContainerMethode bleiben willst, dann schreibe doch s auch in den Container und starte dort das Callback:
Delphi-Quellcode:
type
  TContainer=class(TSimpleRWSync) //oder eben: TContainer=class(TMREWSync)
  private
    FExpData: TExpData; //Besser: TStringList
    Fs:string; //blöder kurzer Name ;-)
   
    function getExpData(i: Integer): String;
    procedure setExpData(i:Integer; const Value: String);
    function getExpDataCount: Integer;

    procedure sets(s:string);
    function gets:string;
    //und hier die Methoden für den Callback wie in dem Beispiel im letzten Thread

  public
    procedure appendExpData(Value:string);
    property ExpData[i: Integer]: String read getExpData write setExpData;
    property ExpDataCount: Integer read getExpDataCount;
    property s:string read....
  end;

sakura 25. Feb 2008 09:58

Re: Via property auf Array zugreifen (Threadsicher?)
 
Ganz wichtig: vergiß die try...finally...end Blöcke nicht, sonst dürfte es ganz schnell Dead-Locks geben, wenn es mal zu einer Exception kommt und ich glaube nicht, dass Du das willst ;)

...:cat:...

TheMiller 25. Feb 2008 10:19

Re: Via property auf Array zugreifen (Threadsicher?)
 
die Try-Except Blöcke werde ich noch einfügen. Was muss beim Except rein? Was spezielles, oder einfach nur eine Fehlerbehandlung bzw. gar nix?

Ein Array habe ich genommen, weil ich es schöner und eigentlich Professioneller finde. Aber wenn es langsamer wird. Ich erwarte viele Werte. Muss ich also nochmal überdenken.

Den Index MUSSTE ich definieren, da sonst immer die Meldung "Inkompatible Typen" in der Setter-Methode aufkamen. Ohne diese Variable kann nicht komipiliert werden. Warum weis ich auch nicht.

Die Message habe ich gesendet, da ich eine Variable mit übergeben muss um die Anwendung erkennen zu lassen, welche Operation sie mit den Daten ausführt. Also eine Case-Of. Oder soll ich da besser noch eine Variable bspw. DataType definieren und dann darüber die Case-Of laufen lassen?

sakura 25. Feb 2008 10:21

Re: Via property auf Array zugreifen (Threadsicher?)
 
Zitat:

Zitat von DJ-SPM
die Try-Except Blöcke werde ich noch einfügen. Was muss beim Except rein?

Hat keiner was von EXCEPT gesagt ;)
Delphi-Quellcode:
procedure TContainer.setExpData(i: Integer; const Value: String);
begin
  BeginWrite;
  try
    SetLength(FExpData, Succ(Length(FExpData)));
    FExpData[High(FExpData)]:=Value; <- Warum muss ich einen Index haben? 
  finally
    EndWrite;
  end;
end;
So als Beispiel ;)

...:cat:...

sakura 25. Feb 2008 10:24

Re: Via property auf Array zugreifen (Threadsicher?)
 
Zitat:

Zitat von DJ-SPM
Den Index MUSSTE ich definieren, da sonst immer die Meldung "Inkompatible Typen" in der Setter-Methode aufkamen. Ohne diese Variable kann nicht komipiliert werden. Warum weis ich auch nicht.

Warum? Einfach, Du deklarierst eine Array-Eigenschaft mit Index und die Getter- und Setter-Methoden müssen zu dieser Deklaration kompatibel sein. Ich würde es wie folgend ändern:
Delphi-Quellcode:
type
  TContainer=class(TSimpleRWSync)
  private
    function getExpData(i: Integer): String;
  public
    procedure AddExpData(const Value: String);

    property ExpData[i: Integer]: String read getExpData;
  end;
Der Rest ist nicht aufgeführt ;)

...:cat:....

TheMiller 25. Feb 2008 10:29

Re: Via property auf Array zugreifen (Threadsicher?)
 
Dankesehr.

Mit dem "Except" habe ich einfach zu schnell gelesen. Das leuchtet auch eher ein - die Zugriffe müssen ja auf jeden Fall wieder freigegeben werden. War dumm von mir!

Nunja, das mit dem Array hat sich demnach auch erledigt. Ich werde wohl doch eine StringList benutzen (obwohl ich arrays schöner finde). Ist doch bei sehr vielen Daten besser, oder?

Danke!

mimi 25. Feb 2008 10:39

Re: Via property auf Array zugreifen (Threadsicher?)
 
Zitat:

Den Index MUSSTE ich definieren, da sonst immer die Meldung "Inkompatible Typen" in der Setter-Methode aufkamen. Ohne diese Variable kann nicht komipiliert werden. Warum weis ich auch nicht.
Nicht umbedingt, denn du könntest auch einfach hinter der Metode Overload; schreiben.

Das sind ja zwei unterschiedliche Aufruf Parameter. Dann geht das.

sirius 25. Feb 2008 12:22

Re: Via property auf Array zugreifen (Threadsicher?)
 
Zitat:

Zitat von DJ-SPM
Den Index MUSSTE ich definieren, da sonst immer die Meldung "Inkompatible Typen" in der Setter-Methode aufkamen. Ohne diese Variable kann nicht komipiliert werden. Warum weis ich auch nicht.

Ach ja, du willst ja immer hinten anhängen und einen bestimmten Array-Wert lesen. Das geht innerhalb eines Propertys nicht (da habe ich ja auch einen Fehler oben - Berichtigung folgt sogleich). Die Lösung haben meine Vorredner ja schon gezeigt.

TheMiller 25. Feb 2008 12:59

Re: Via property auf Array zugreifen (Threadsicher?)
 
Hm stimmt.

Jetzt würde ich das Array doch gerne lassen... Wie viel langsamer ist denn ein Array gegenüber einer Stringliste und ab wie vielen Einträgen macht sich das denn bemerkbar?

mimi 25. Feb 2008 16:40

Re: Via property auf Array zugreifen (Threadsicher?)
 
Ich würde sagen das ist unterschiedlich je nach Auslastung der CPU und die Leistung der CPU.

Aber so bis 500(kann auch mehr oder weniger sein) Einträge dürftest du nix merken, Kommt auch drauf an wie groß dein Record/Klass ist...

TheMiller 25. Feb 2008 17:09

Re: Via property auf Array zugreifen (Threadsicher?)
 
Dann lass ich es einfach so wie's ist und im Produktiveinsatz kann man ja eine Testversion mit einer Stringlist machen.

Ich danke euch!

mimi 25. Feb 2008 17:16

Re: Via property auf Array zugreifen (Threadsicher?)
 
Schau dir doch noch die TList und die TObjectList an. Die werden auch gerne genommen. Evlt. währe auch eine "Doppelt Verkette Liste" Etwas für dich, da habe ich mal für mich eine Klasse geschrieben, die so aufgebaut ist wie die TObjectList.

TheMiller 25. Feb 2008 20:16

Re: Via property auf Array zugreifen (Threadsicher?)
 
Zitat:

Zitat von mimi
du könntest auch einfach hinter der Metode Overload; schreiben.

Nein, da kommt die Fehlermeldung:

Delphi-Quellcode:
Methode setExpData nicht in Basisklasse enthalten
Und nun?

mimi 25. Feb 2008 20:50

Re: Via property auf Array zugreifen (Threadsicher?)
 
poste noch mal den Header von der Klasse wo du overload verwendest und den Conde wo die Fehler Meldung auftritt. Eigentlich sollte das Klappen du solltest setExpData evlt. als virtual Definieren. Zeigt doch mal deine Bases Klasse(oder ist das eine Standard Borland Klasse? Dann natürlich nicht*G*)

TheMiller 25. Feb 2008 20:53

Re: Via property auf Array zugreifen (Threadsicher?)
 
Sorry Leute, ich war nur zu dumm. Hab vergessen bescheid zu sagen... Hat funktioniert!


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