AGB  ·  Datenschutz  ·  Impressum  







Anmelden
Nützliche Links
Registrieren
Zurück Delphi-PRAXiS Sprachen und Entwicklungsumgebungen Object-Pascal / Delphi-Language Delphi TList in Thread synchronisieren (CriticalSection)
Thema durchsuchen
Ansicht
Themen-Optionen

TList in Thread synchronisieren (CriticalSection)

Offene Frage von "Zacherl"
Ein Thema von Zacherl · begonnen am 2. Feb 2010 · letzter Beitrag vom 2. Feb 2010
Antwort Antwort
Benutzerbild von Zacherl
Zacherl

Registriert seit: 3. Sep 2004
4.629 Beiträge
 
Delphi 10.2 Tokyo Starter
 
#1

TList in Thread synchronisieren (CriticalSection)

  Alt 2. Feb 2010, 11:03
Hey,

ich habe ein kleines Problem. Und zwar habe ich eine von TThread abgeleitete Klasse, welche in einer Schleife Daten aus einer Liste bearbeitet:
Delphi-Quellcode:
while not Terminated do
begin
  for I := 0 to FList.Count - 1 do
  begin
    // blabla ..
  end;
end;
Der Thread hat entsprechende Methoden, um Daten zur Liste hinzuzufügen, zu entfernen oder zu verschieben. Jetzt habe ich ein paar Fragen:
Muss ich hier die Zugriffe überhaupt synchronisieren oder regelt TList das intern?
Wenn ja, wie mache ich das am geschicktesten?

Ich hatte mal testweise ein privates Feld CS: TCriticalSection angelegt, welches ich folgendermaßen innerhalb des Threads verwende:
Delphi-Quellcode:
while not Terminated do
begin
  CS.Enter;
  for I := 0 to FList.Count - 1 do
  begin
    // blabla ..
  end;
  CS.Leave;
end;
Die Methoden zum Ändern der Liste wurden entsprechend mit Enter und Leave geklammert.
Das Problem war nun, dass beim Aufruf von z.B. AddData() der Main Thread dauerhaft blockiert wurde, die CriticalSection im Thread also scheinbar nie released wurde.

Hat jemand da paar Tips für mich? Achso der Thread ist zeitkritisch. Einbauen von Sleep oder Ähnlichem ist hier keine Lösung.

Viele Grüße
Zacherl
  Mit Zitat antworten Zitat
Astat

Registriert seit: 2. Dez 2009
Ort: München
320 Beiträge
 
Lazarus
 
#2

Re: TList in Thread synchronisieren (CriticalSection)

  Alt 2. Feb 2010, 11:10
Hallo Zacherl, wenn die TList nur intern im Thread verwendet wird, brauchst Du keine Synchronisation.
Sobald ein anderer Thread auf diese Liste zugreift, musst Du synchronisieren.
Für soetwas gibt es die TThreadList (sollte in Delphi auch so heissen)

lg. Astat
Lanthan Astat
06810110811210410503210511511603209711003210010110 9032084097103
03211611111604403209711003210010110903210010510103 2108101116122
11610103209010110510810103206711110010103210511003 2068101108112
10410503210310111509910411410510109810111003211910 5114100046
  Mit Zitat antworten Zitat
Benutzerbild von Zacherl
Zacherl

Registriert seit: 3. Sep 2004
4.629 Beiträge
 
Delphi 10.2 Tokyo Starter
 
#3

Re: TList in Thread synchronisieren (CriticalSection)

  Alt 2. Feb 2010, 11:16
Ah vielen Dank, diese TThreadList sieht gut aus. Da die Methoden wie AddData(), etc. vom Main Thread aus aufgerufen werden sollen, nehme ich mal an, dass eine Synchronisation notwendig sein wird.
  Mit Zitat antworten Zitat
Astat

Registriert seit: 2. Dez 2009
Ort: München
320 Beiträge
 
Lazarus
 
#4

Re: TList in Thread synchronisieren (CriticalSection)

  Alt 2. Feb 2010, 11:26
Zitat von Zacherl:
Ah vielen Dank, diese TThreadList sieht gut aus. Da die Methoden wie AddData(), etc. vom Main Thread aus aufgerufen werden sollen, nehme ich mal an, dass eine Synchronisation notwendig sein wird.
Ja, mit LockList Liste sperren, dann damit arbeiten.

lg. Astat
Lanthan Astat
  Mit Zitat antworten Zitat
Benutzerbild von Zacherl
Zacherl

Registriert seit: 3. Sep 2004
4.629 Beiträge
 
Delphi 10.2 Tokyo Starter
 
#5

Re: TList in Thread synchronisieren (CriticalSection)

  Alt 2. Feb 2010, 11:29
Danke dir, ich probier mal ob das dann besser funktioniert als mit TCriticalSection
  Mit Zitat antworten Zitat
Benutzerbild von himitsu
himitsu

Registriert seit: 11. Okt 2003
Ort: Elbflorenz
44.033 Beiträge
 
Delphi 12 Athens
 
#6

Re: TList in Thread synchronisieren (CriticalSection)

  Alt 2. Feb 2010, 11:42
Jupp, TList ist nicht threadsicher.

(Wenn auf irgendwas nur innerhalb irgendeines Threads und nur in diesem, zugegriffen wird, dann muß man nichts absichern)


PS: um Daten einer Liste abzuarbeiten, hatt ich mir mal dieses gebastelt
http://www.delphipraxis.net/internal...t.php?t=167796

Zitat:
Das Problem war nun, dass beim Aufruf von z.B. AddData() der Main Thread dauerhaft blockiert wurde, die CriticalSection im Thread also scheinbar nie released wurde.
erstmal der Resourcen-Schutzblock:
Delphi-Quellcode:
while not Terminated do
begin
  CS.Enter;
  try
    for I := 0 to FList.Count - 1 do
    begin
      // blabla ..
    end;
  finally
    CS.Leave;
  end;
end;
Denn wenn etwas passiert (z.B. eine Exception), dann wird da wirklich kein .Leave aufgerufen.

Und dann rufst du praktisch direkt nach dem .Leave fast sofort wieder .Enter auf.
Wenn also hier Windows nicht rein zufällig an dieser Stelle eine Pause einlegt und den anderen Threads so überhaupt die Möglichkeit gibt hier einzusteigen, dann kann auch nie ein anderer Thread an der Stelle die CS "übernehmen".

Delphi-Quellcode:
while not Terminated do
begin
  CS.Enter;
  try
    for I := 0 to FList.Count - 1 do
    begin
      // blabla ..
    end;
  finally
    CS.Leave;
  end;
  Sleep(0);
end;
0 gibt nur für einen "Zyklus" die Kontrolle ab,
so daß die anderen Threads überhaupt 'ne Chance bekommen.
Unter Umständen könnte hier aber auch eine noch längere Pause hilfreich sein.
Garbage Collector ... Delphianer erzeugen keinen Müll, also brauchen sie auch keinen Müllsucher.
my Delphi wish list : BugReports/FeatureRequests
  Mit Zitat antworten Zitat
Benutzerbild von Zacherl
Zacherl

Registriert seit: 3. Sep 2004
4.629 Beiträge
 
Delphi 10.2 Tokyo Starter
 
#7

Re: TList in Thread synchronisieren (CriticalSection)

  Alt 2. Feb 2010, 12:20
Danke dir für deine ausführliche Antwort. Deinen Stack werde ich mir mal ansehen, das sieht interessant aus Der kleine Code eben war nur ein Beispiel. Im richten Programm hatte ich natürlich mit try .. finally gearbeitet. Mit der ThreadList funktioniert es bisher auch ohne Sleep() relativ gut. Manchmal muss der Main Thread allerdings schon einige Zeit warten. Ich teste das ganze mal mit einem Sleep(0) oder Sleep(1). Das könnte von der Performance her grade noch vertretbar sein.

Edit: Mit Sleep(0) sieht es ziemlich perfekt aus
  Mit Zitat antworten Zitat
Benutzerbild von himitsu
himitsu

Registriert seit: 11. Okt 2003
Ort: Elbflorenz
44.033 Beiträge
 
Delphi 12 Athens
 
#8

Re: TList in Thread synchronisieren (CriticalSection)

  Alt 2. Feb 2010, 12:37
Zitat von Zacherl:
Edit: Mit Sleep(0) sieht es ziemlich perfekt aus


Zitat von Zacherl:
Deinen Stack werde ich mir mal ansehen, das sieht interessant aus
Nur D2005 und Generics könnte etwas problematisch werden
Garbage Collector ... Delphianer erzeugen keinen Müll, also brauchen sie auch keinen Müllsucher.
my Delphi wish list : BugReports/FeatureRequests
  Mit Zitat antworten Zitat
Benutzerbild von Zacherl
Zacherl

Registriert seit: 3. Sep 2004
4.629 Beiträge
 
Delphi 10.2 Tokyo Starter
 
#9

Re: TList in Thread synchronisieren (CriticalSection)

  Alt 2. Feb 2010, 12:50
Zitat von himitsu:
Nur D2005 und Generics könnte etwas problematisch werden
Oh das benutze ich zum Glück schon länger nicht mehr Hab nur mein Profil lange nicht aktualisiert.
  Mit Zitat antworten Zitat
Antwort Antwort


Forumregeln

Es ist dir nicht erlaubt, neue Themen zu verfassen.
Es ist dir nicht erlaubt, auf Beiträge zu antworten.
Es ist dir nicht erlaubt, Anhänge hochzuladen.
Es ist dir nicht erlaubt, deine Beiträge zu bearbeiten.

BB-Code ist an.
Smileys sind an.
[IMG] Code ist an.
HTML-Code ist aus.
Trackbacks are an
Pingbacks are an
Refbacks are aus

Gehe zu:

Impressum · AGB · Datenschutz · Nach oben
Alle Zeitangaben in WEZ +1. Es ist jetzt 23:50 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